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从 Web 项 目 开发 的 角度 出 发 ， 打 造 点 、 线 、 面 一 体 化 的 学 习 架 构 ， 让 读者 既 能 够 学 习 到 子 功能 模块 的 实现 方 
法 、 开 发 技巧 ， 同 时 又 能 够 合理 地 运用 子 功 能 模块 ， 打 造 出 完整 的 项 目 。 

本 书 模块 涉及 面 广 ， 实 用 性 强 。 通 过 对 本 书 的 学 习 ， 读 者 可 以 了 解 Web 开发 中 各 个 环节 的 特点 、 功 能 ， 以 
及 如 何 合理 地 连接 各 个 环节 ， 打 造 出 完整 的 项 目 。 另 外 ， 本 书 中 所 提供 的 模块 不 仅 包 括 常用 功能 模块 ， 还 包含 
目前 比较 流行 的 热点 模块 ， 既 可 达到 读者 二 次 开发 的 目的 ， 同 时 亦 可 满足 读者 对 流行 、 热 点 内 容 的 好 奇 心 。 

本 书 提供 了 大 量 的 自学 视频 、 源 程序 、 素 材 ， 提 供 了 相关 的 模块 库 、 案 例 库 、 素 材 库 、 题 库 等 多 种 形式 
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适合 作为 相关 培训 机 构 的 师 生 和 软件 开发 人 员 的 参考 资料 。 
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众 里 寻 她 千百度 ， 暮 然 回首 ， 那 人 却 在 灯火 阑珊 处 ! 

每 个 想 要 成 为 程序 员 的 、 已 经 是 初级 程序 员 的， 或 者 已 经 是 中 级 程序 员 的 读者 ， 都 有 
这 样 一 个 梦想 ， 期 待 着 自己 有 一 天 能 够 成 为 一 名 高 级 程序 设计 大 师 。 

每 个 人 都 在 为 这 个 理想 而 奋斗 着 ， 不 断 地 付出 ， 开 发 着 各 种 各 样 的 项 目 、 模 块 ， 以 此 
来 积累 经 验 ， 提 高 技术 水 平 。 当 有 一 天 梦想 实现 的 时 候 ， 回 想起 自己 是 如 何 做 到 的 ， 慕 然 
发 现 正 是 那些 各 种 各 样 的 项 目 、 模 块 成 就 了 自己 。 

正 所 谓 ， 众 里 寻 她 千百度 ， 暮 然 回 首 ， 那 人 却 在 灯火 阑珊 处 ! 

本 书 精 选 了 15 个 模块 ， 直 接 将 这 灯火 阑珊 处 的 “她 ”呈现 给 诸位 读者 ， 希 望 读者 能 
够 早日 成 就 梦想 。 


本 书 内 容 


本 书 分 为 15 章 ， 包含 80 项 开发 技术 、 经 验 , 涵盖 注册 登录 、 分 页 显示 、 上 传 与 下 载 、 
网 页 计数 器 、 购 物 车 、 在 线 支付 、 假 日 公告 管理 、 投 票 系统 、 留 言 本 、 在 线 论坛 、 博 客 、 
在 线 编程 词典 、 邮 件 发 送 系 统 、FTP 管理 系统 和 创建 PDF 文档 等 多 个 模块 的 内 容 。 所 选 模 
块 均 来 源 于 一 线 开发 人 员 的 第 一 手 资料 ， 各 个 模块 在 实际 开发 中 所 占 的 比重 如 下 图 所 示 。 


本 书 特色 


回 ”纵向 延伸 。 本 书 精 选 了 15 个 模块 ， 涵 盖 项 目 开发 中 应 用 的 各 方面 技术 和 数据 库 ， 
体现 了 从 小 到 大 、 从 易 到 难 的 学 习 过 程 。 
回 ”横向 连接 。 属 于 PHP 系列 图 书 的 一 部 分 ， 是 PHP 入 门 学 习 、 范 例 练 习 的 延伸 ， 以 


义 和 相 关 技 术 、 技 巧 。 


或 应 用 。 
| 本 书 配套 资源 


了 tp 六 并 要 志 煤 角 


模块 开发 的 形式 锻炼 读者 Web 开发 的 能 力 ， 并 且 逐 步 向 更 高 级 别 的 项 目 开 发 迈进 。 
易学 、 易 用 。 本 书 中 的 每 个 模块 都 通过 “技术 提炼 ”栏目 ， 对 代码 中 重要 的 对 象 、 
方法 、 语 句 和 重点 知识 等 进行 说 明 ， 以 便于 读者 在 阅读 代码 时 透彻 理解 代码 的 含 


赠送 所 有 模块 源 代码 。 书 中 所 有 模块 均 提 供 源 代 码 ， 用 户 在 开发 中 可 以 快速 借鉴 


回 提供 72 集 ( 节 ) 视频 录像 。 本 书 提 供 了 72 集 ( 节 ) 视频 录像 ， 读 者 可 以 边 看 边 


| 本 书 提供 了 内 容 丰 富 的 配套 资源 ， 包 括 自学 视频 、 源 程序 、 素 材 ， 以 及 模块 库 、 案 例 
| 库 、 题 库 、 素 材 库 等 多 项 辅助 内 容 ， 读 者 朋友 可 以 通过 如 下 方式 获取 。 


第 1 种 方式 : 


| (1) 登录 www.tup.com.cn， 在 网 页 右上 角 的 搜索 文本 框 中 输入 本 书 书 名 注意 区 分 
| 大 小 写 和 留 出 空格 )， 或 者 输入 本 书 关键 字 ， 或 者 输入 本 书 ISBN 号 (注意 去 掉 ISBN 号 间 


| 隔 线 “-”)， 单 击 “ 搜 索 ” 按 钮 。 
(2) 找到 本 书后 单 击 超 链接 ， 在 该 书 的 网 页 下 侧 单 击 “ 网 络 资源 ” 超 链接 ， 即 可 下 载 。 


第 2 种 方式 : 


配套 资源 目录 如 下 图 所 示 。 


Bn 


配套 资源 一 


访问 本 书 的 新 浪 微 博 PHPbook， 找 到 配套 资源 的 链接 地 址 进行 下 载 。 
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团 bottom .php 
@ changepub. php 
转 downcode. php 
团 downfile. php 


加 ] 程 序 使 用 说 明 . doe 


Bn 名 ( 购 物 车 功能 模块 ,exe 
a 了 (类 的 圭 装 与 实例 化 .exe 
有 商品 展示 模块 .exe 

人 1 首页 设计 .exe 
加 了 i 数据 库 的 设计 .exe 


甘 用 户 注册 与 登录 . exe 


本 书 约定 


项 目 使 用 方法 
在 学 习 本 书 的 过 程 中 ， 可 以 从 配套 资源 中 复制 模块 ， 去 掉 其 只 读 属性 。 有 些 模块 需要 | 
使 用 相应 的 数据 库 或 第 三 方 资源 ， 此 类 模块 在 使 用 前 需要 进行 相应 配置 ， 详 细 使 用 方式 参 
见 配套 资源 中 的 使 用 说 明 。 此 外 ， 如 果 直 接 将 本 书 内 容 用 于 商业 用 途 ， 由 此 产生 的 不 良 后 | 
果 由 用 户 自己 承担 。 
回 “部 分 项 目 只 给 出 关键 代码 
由 于 篇 幅 限制 ， 书 中 有 些 模块 只 给 出 了 关键 代码 ， 完 整 代码 参见 配套 资源 中 的 对 应 程序 。| 


读者 人 群 
本 书 非常 适合 以 下 人 员 阅 读 : 
从 事 PHP 编程 行业 的 开发 人 员 
有 一 定语 言 基础 ， 想 进一步 提高 技能 的 人 员 
大 中 专 院 校 的 老师 和 学 生 
即将 走 上 工作 岗位 的 大 学 毕业 生 
相关 培训 机 构 的 老师 和 学 员 
PHP 编程 爱好 者 


读者 服务 & 本 书 勘误 


读者 在 使 用 本 书 过 程 中 遇 到 的 所 有 问题 ， 均 可 通过 以 下 方式 联系 我 们 。 

1. 新 浪 微 博 : PHPbook。 

及 时 发 布 读者 答疑 、 本 书 勘误 、 配 套 资料 更 新 等 内 容 。 

2. 腾讯 QQ: 4006751066。 | 

3. 登录 网 站 : www.mingribook.com， 在 论坛 、 勘 误 发 布 、 读 者 纠 错 、 技 术 支持 、 读 | 
者 之 家 等 栏目 中 的 相关 模块 中 提问 、 留 言 或 查看 。 | 


本 书 作 者 


本 书 由 明日 科技 组 织 编写 ， 参 与 编写 的 有 潘 凯 华 、 李 慧 、 刘 欣 、 高 春 攀 、 王 小 科 、 王 | 
军 、 赵 会 东 、 董 大 永 、 尹 强 、 李 继 业 、 张 舌 、 赛 奎 春 、 宋 坤 、 刘 锐 宁 、 梁 水 、 李 伟 明 、 刘 | 
彬 彬 、 安 剑 、 孙 秀 梅 、 巩 建华 、 房 大 伟 、 吕 双 、 刘 云 峰 、 王 国 辉 、 李 钟 尉 、 陈 丹 丹 、 卢 翰 、 | 
梁 晓 岗 、 杨 丽 、 顾 丽 丽 、 刘 龄 龄 、 王 乐 乐 、 陈 英 、 李 贺 、 朱 晓 、 肖 和 狗 、 李 狗 和 刘 冠 男 等 。 | 
由 于 作者 水 平 有 限 ， 玻 漏 和 不 足 之 处 在 所 难免 ， 敬 请 广大 读者 朋友 批评 指正 。 
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注册 登录 模块 
( 名 自学 视频 、 源 程序 ; 配套 资源 mA1\) 


从 动态 网 站 诞生 之 初 ， 注 册 登 录 模 块 就 成 为 网 站 必 不 可 少 的 组 成 部 分 ， 并 
且 以 其 功能 单一 、 实 现 相对 简单 的 将 点 ， 成 为 入 门 级 的 模块 。 随 着 Web 2.0 的 
发 展 ， 人 机 交互 成 为 主流 趋势 ， 注 册 登 录 模 块 也 顺应 了 时 代 的 发 展 ， 变 得 更 加 
有 “亲和力 ”， 用 户 那 种 花 了 许久 时 间 填 写 注册 表单 ， 提交 后 却 发 现 信息 填写 错 
误 ， 需 要 重新 填写 的 惨痛 经 历 越 来 越 少 ， 随 着 Ajax 技术 的 兴起 ， 这 种 情况 就 更 
少 发 生 了 。 本 章 将 开发 一 个 功能 齐全 并 有 着 良好 交互 性 的 注册 登录 模块 。 通 过 
本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


WI 注册 信息 实时 提示 
MW E-mail 激活 技术 
Pl GD2 验证 码 

WI 更 换 验 证 码 

WI 防 5QL 注入 技术 


Note 


a 人 tpg 量 扩 类 持 角 


1.1 注册 及 登录 验证 概述 


注册 及 登录 验证 模块 (以 下 简称 注册 登录 模块 ) 的 功能 比较 单一 ， 除 了 用 于 验证 用 户 
| 信息 的 有 效 性 和 一 些 必 要 的 安全 设置 外 ， 更 多 的 是 解决 如 何 提高 程序 的 可 操作 性 ， 以 更 好 
”地 方便 用 户 。 注 册 登 录 模块 主要 由 用 户 注 册 、 用 户 登 录 和 找 回 密码 3 部 分 组 成 。 


| 1.1.1 用 户 注册 流程 


| 用户 注 册 的 信息 分 为 必 填 信息 和 可 选 信息 。 必 填 信息 默认 是 可 见 的 ， 而 可 选 信息 默认 
| 是 隐藏 的 。 在 必 填 信息 不 完整 的 情况 下 ,“ 注 册 ” 按 钮 是 不 可 用 的 。 对 于 可 选 信息 ， 用 户 
”可 以 选择 填写 或 不 填写 。 但 如 果 填写 了 ， 那么 就 需要 对 信息 的 正确 性 进行 检验 。 当 注册 成 
| 功 后 ， 系 统 将 向 用 户 所 填写 的 E-mail 发 送 一 封 激活 邮件 ， 用 户 只 有 激活 后 , 才 可 使 用 该 账 
| 号 进行 登录 。 用 户 注册 流程 图 如 图 1.1 所 示 。 


- - 雪 显示 非 必 填 信息 上 - -叶轮 入 信息 


| 图 1.1 用 户 注 册 流 程 图 

1.12 用 户 登 录 流程 

用 户 登录 时 ， 除 了 需要 用 户 名 和 密码 外 ， 还 需要 填写 随机 生成 的 验证 码 。 当 验证 码 显 
| 示 不 清楚 时 ， 可 以 通过 超 链 接 进行 更 换 而 无 须 重新 刷新 页 面 。 登 录 处 理 对 登录 进行 了 一 些 


限制 。 例 如 ， 输 入 3 次 无 效用 户 名 后 ， 将 使 用 Cookie 技术 禁止 该 用 户 继续 登录 ; 当 用 户 
”连续 3 次 输入 错误 密码 后 ， 该 账号 将 被 冻结 。 用 户 登录 流程 如 图 1.2 所 示 。 


J ss 
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图 1.2 用 户 登 录 流程 图 
1.1.3” 找 回 密码 流程 


当 用 户 因为 个 别 原因 忘记 密码 时 ， 可 以 通过 密码 保护 问题 及 答案 来 找 回 。 如 果 用 户 输 
入 正确 ， 那 么 系统 会 将 密码 通过 电子 邮件 的 形式 发 送 到 用 户 的 注册 邮箱 中 。 如 果 没 有 填写 | 
密码 保护 问题 及 答案 或 者 回答 错误 ， 则 无 法 找 回 。 找 回 密码 的 流程 如 图 1.3 所 示 。 | 


图 1.3 找 回 密码 流程 图 


~ WA CE 


1.1.4 程序 预览 


注册 登录 模块 主要 由 两 个 页 面 组 成 ， 即 注册 页 面 和 登录 页 面 。 注 册页 面 与 登录 页 面 的 
运行 效果 如 图 1.4 和 图 1.5 所 示 。 
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这 机 下 载 。 在 盐 订 丙 


如 果 培 没有 注册 ， 记 悠 广 册 后 本 录 。 扣 果 您 已 经 注册 ， 刘 右 记 了 守 码 ， 窗 可 以 通过 
找 加 更 三 村 您 且 杰 码 防 回 - 


出 生日 期 。 |1999-12- 人 1 
联系 电话 |0431-649777T 


| QQ 号 码 : |315527355 


Ey Bg En 


技术 服务 热 录 ; 0- BTS-1068 传真 ; 0431-B4938TTY 企业 孝 补 :nrycxdbmrbecd son 
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图 1.4 注册 页 面 


加 


首页 用 阳 疡 管理 产品 介绍 


如 果 您 没有 注册 ， 请 您 注册 后 再 登录 。 如 果 您 已 经 注册 ,但 志 记 了 黎 码 ， 您 可 以 通过 
找 回 密码 将 您 的 客 码 找 回 . 


登录 和 名称 : mr 


要 录 帮 码 : |eeeeee 
5c ] 65b < 看 再 


验证 玛 : 


EE Gg Ea 


技术 服务 热 贱 : 400-675-1066 传真 0431-64939T77 企业 岂 箱 : arbe cdBarbeed eon 
公司 地 址 : 长 春 市 亚 奉 广 声 [ 座 ” 吉 ICP 备 125689 
Copyrieht Or nrbeed co ALl 了 起 ts Reserveal 


图 1.5 登录 页 面 


1.2 数据 库 设计 


本 模块 应 用 MySQL 数据 库存 储 用 户 注册 信息 ， 有 关 MySQL 数据 库 及 数据 表 的 创建 


。4。 
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将 在 本 节 进行 详细 讲解 。 
1.2.1 ”数据 库 设计 概述 


实现 注册 登录 模块 使 用 的 数据 库 是 db_reglog， 其 中 包含 一 个 数据 表 由 member, 该 表 | 倒 ] 
中 有 12 个 字段 。tb_member 表 的 结构 及 说 明 如 图 1.6 所 示 。 


Note 
reglog 四 要 thmember | 
唱 齐 蜗 。 因 结构“ 允 sal 万 机 过 计生 入。 医 导 出 。 箱 Impor 多 作 位 稍 宁 男 担 除 | 
有 站 整理 县 Null 可 额外 训 明 ! 
性 认 ! 
[a In 百 auto_incrament 如 叶 | 
name Varchar0po) gb2312-chinese-ci 否 用 户 各 称 | 
Tm password varcher(200) gb2312-chinese-ci 理 用 户 窗 码 | 
TF answer varchar0oo) gb2312.chinese.ci 否 密 辽 区 守 | 
厂 quesuon varcharnoo) gbz3lz.chinese.cl 吾 光电 是 | 
Femail varcharf 00) gb2312_chinese ci 可 ! 
厂 realname varchar(200) gb2312_chinese-cl 理 革 好 限制 ! 
rm birhday date 亚 出 生日 其 | 
厂 telephone varchar(20) 9b2312_chinese_cl 亚 号码 ! 
Faq varchartl 3) 。 gb2312_chinese_cl 可 i ! 
Tm count in 壮 到 | 区 壮阳 制 | 
Factive im 了 时 活 ! 


图 1.6 tb_member 表 结 构 及 说 明 | 
1.2.2 ”封装 数据 库 连 接 、 操 作 类 


为 了 便于 维护 ， 减 少 代码 匈 余 ， 本 模块 使 用 了 一 个 简单 的 自 定义 数据 库 类 。 下 面 给 出 
类 中 的 主要 代码 。 

首先 给 出 的 是 类 中 的 成 员 变 量 ， 其 中 $host、$name、$pwd 和 $dBase 变量 是 类 初始 化 
时 所 使 用 的 ， 有 默认 值 ， 其 他 变量 都 是 相关 的 数据 信息 。 代 码 如 下 : 


01 <?php | 
02 classopmysql{ | 
03 Private $host = "localhost'; // 服 务 器 地 址 | 
04 Private $name = 'root'; // 登 录 账 号 | 
05 private $pwd = "111'; // 登 录 密 码 | 
06 Private $dBase = 'db_reglog'; // 数 据 库 名 称 | 
07 Private $conn ="; /数据 库 连 接 资 源 | 
08 Private $result = "; /结果 集 | 
09 private $msg = "; // 返 回 结果 | 
10 Private $fields; /返回 字段 | 
11 Private $fieldsNum = 0; // 返 回 字段 数 | 
12 Private $rowsNum = 0; // 返 回 结果 数 | 
13 Private $rowsRst = "; // 返 回 单条 记录 的 字段 数组 

14 private $filesArray = array(); /返回 字段 数组 | 
15 private $rowsArray = array(); // 返 回 结 果 数 组 | 


接 下 来 给 出 的 是 构造 函数 。 该 构造 函数 包含 4 个 参数 ， 这 4 个 参数 都 可 以 被 省 略 而 使 


用 默认 值 。 参 数 处理 后 ， 调 用 init_ conn(0) 函 数 。 代 码 如 下 : 
01 function construct($host=",$name=".$pwd=",$dBase="){ 
02 1f($host !=") 


.5°. 
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03 Sthis->host = $host: 

04 if($name !=") 

05 Sthis->name = $name; 
06 if($pwd (=") 

07 Sthis->pwd = $pwd; 

08 if($dBase !=") 

09 Sthis->dBase = $dBase; 
10 S$this->init_conn(); 

11 } 


init_conn() 函 数 : 根据 成 员 变 量 中 的 值 来 创建 数据 库 连 接 源 。 代 码 如 下 : 


01 
02 
03 
04 
05 
06 


// 连 接 数据 库 
function init_connO{ 

S$this->conn=(Wmysql connect($this->host,$this->name,$this->pwd); 

@mysql select db($this->dBase.$this->conn); 

mysql_query("set names gb2312"); // 设 置 编码 
} 


下 面 


始 就 是 类 中 的 操作 函数 了 。 首 先 介 绍 查 询 函 数 。 查 询 函 数 根据 传 过 来 的 SQL 


语句 进行 查询 ， 并 将 查询 结果 保存 到 成 员 变量 $result 中 。 符 号 “@” 的 作用 是 屏蔽 错误 信 


息 。 代 码 恕 


01 
02 
03 


1 下 : 
// 查 询 结果 
function mysql query_rst($sqD){ 

if($this->conn =— "){ 

Sthis->init_conn(); 

1 

Sthis->result = (Wmysql_query($sql,$this->conn); 
} 


返回 查询 记录 数 函 数 : 根据 查询 结果 ， 返 回 记 录 数 。 代 码 如 下 : 


function getRowsNum($sqD{ 
$this->mysql_ query_rst($sqD): 
这 mysqlL ermo0 一 0){ 
retum mysql num rows($this->result); 
}else{ 
return ": 
上 
} 


| 取得 记录 数组 函数 : 将 查询 结果 输出 成 一 个 数组 并 返回 。 该 函数 处 理 的 是 单条 记录 。 
| 代码 如 下 : 


01 
02 


function getRowsRst($sqD){ 
$this->mysql query_rst($sq]); 
这 mysql ermror0 一 0){ 
Sthis->rowsRst = mysql fetch array($this->result MYSQL ASSOO); 
return $this->rowsRst:; 
}else{ 


.6°. 


01 
02 
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return "; 


取得 记录 数组 函数 : 功能 同上 ，, 只 是 这 里 返回 一 个 有 多 条 记录 的 二 维 数组 。 代码 如 下 : | 


// 取 得 记录 数组 (多 条 记录 ) 
function getRowsArray($sqD){ 
$this->mysql query_rst($sqD): 
if(mysql ermo0 一 0){ 
while($row = mysql fetch array($this->result MYSQL ASSOC)) { 
Sthis->rowsArray[] = $row: 
} 
return $this->rowsArray; 
}else{ 
return "; 
bp 
} 


返回 更 新 、 删 除 、 添 加 的 记录 数 函 数 : 该 函数 用 来 更 新 、 删 除 、 添 加 记录 并 获取 受 影 
响 的 记录 数 ， 进 而 判断 操作 是 否 成 功 。 代 码 如 下 : 


01 
02 


function uidRst($sqD){ 
if($this->conn — "){ 
Sthis->init_conn(); 
b 
@mysql_query($sq)): 
Sthis->rowsNum = (Wmysql_affected_rows(); 
让 mysql_ermo0 一 0){ 
return $this->rowsNum: 
}else{ 
return "; 
} 
} 


释放 结果 集 函 数 : 将 不 再 使 用 的 数据 删除 ， 释 放 内 存 。 代 码 如 下 : 


08 


function close_rstO{ 
mysql_free_result($this->result); 
Sthis->msg = "; 
Sthis->fieldsNum = 0; 
S$this->rowsNum = 0: 
Sthis->filesArray = 
Sthis->rowsArray ="; 


} 


关闭 数据 库 函 数 : 关闭 数据 库 。 代 码 如 下 : 


01 
02 
03 
04 


// 关 闭 数 据 库 

function close_conn(O{ 
Sthis->close_rst(); 
mysql_close($this->conn): 


wy 
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05 S$this->conn = "; 
06 } 

re: 

08 $conne = new opmysql(); 
09 ?> 


1.3 用 户 注 册 
| 1.3.1 用 户 注册 功能 概述 


用 户 注 册 时 ， 必 填 信 息 包括 用 户 名 、 密 码 和 E-mail。 只 有 这 些 必 填 信息 添加 完整 并 正 
| 确 时 ,“ 注 册 ” 按 钮 才 被 激活 。 这 时 用 户 可 以 选择 注册 或 填写 详细 信息 后 再 进行 注册 。 在 
整个 注册 过 程 中 , 对 于 用 户 的 每 一 次 操作 , 系统 都 将 给 出 友好 的 提示 来 帮助 用 户 完成 注册 。 
| 注册 功能 的 运行 结果 如 图 1.7 所 示 。 


‘ 


这 


加 果 您 没 有 注册 ， 讲 您 注册 后 刺 壮 录 。 加 并 你 已 经 注册 ,胆石 记 了 证 码 ,您 可 以 通过 
找 回 宪 罗 将 沁 的 密码 拷 加 。 


|ePi a! 

注册 寥 码 。 |seeeee 到 于 要 求 。 环 848 度 : 和 
和 正确 

的 入 六 


外 
EE 
或 
EE 
日 
中 | 
8 


qq 号 到 57627356 
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技术 服务 热 柳 : 


图 1.7 注册 功能 的 运行 结果 


| 注册 功能 的 实现 涉及 5 个 核心 文件 :registerphp 〈 注 册 表 单 页 )、register_chk.php 〈 注 
， 册 信息 处 理 页 ).chlkname.php( 检 测 用 户 名 )、registerjs( 注 册 信 息 验证 脚本 ) 和 activation php 
| 激活 处 理 页 )。 


13.2 注册 页 面 设计 


| 在 注册 表单 页 中 , 主要 包含 填写 信息 的 文本 框 、 信 息 提示 的 div 标签 和 3 个 相关 按钮 。 
| 注册 表单 页 中 使 用 的 元 素 如 表 1.1 所 示 。 
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表 1.1 注册 页 面 中 的 主要 元 素 


这 个 函数 的 功能 是 返回 被 触发 的 “id=idValue” 的 元 素 对 象 。 代 码 如 下 : 


function $(idValue){ 
Teturn document.getElementById(idValue); 


} 
接 下 来 是 window.onlaod 事件 ， 表 示 当 窗 体 被 载 入 时 触发 。function0{…} 表 示 当 页 面 
被 载 入 时 所 要 进行 的 操作 。 代 码 如 下 : | 


Wdow onload = finetonOt | 


元 素 名 称 元 素 属性 值 说 明 | 
script "js/xmlhttprequest.js" 载 入 xmlhttprequest 对 象 脚本 ! 
script "js/register.js" 载 入 注册 信息 验证 脚本 | 傣 六 
text id="regname" name="regname" 注册 名 称 | 人 
div id="namediv" 注册 名 称 提示 信息 JNote 
password id="regpwdl" name="regpwd1" 注册 密码 | 
div id="pwddiv1" 密码 提示 信息 | 
password id="regpwd2" name="regpwd2" 确认 密码 | 
div id="pwddiv2" 确认 密码 提示 信息 | 
text id="email" name="email" E-mail | 
div id="emaildiv" E-mail 提示 信息 | 
div id="morediv" style="display:none:" 隐藏 /显示 非 必 填 信息 | 
text id=" question" name="question" 密码 保护 问题 | 
text id="answer" name="answer" 密码 保护 答案 | 
text id="realname" name="realname" 真实 姓名 | 
text id="birthday" name="birthday" 出 生日 期 | 
text id="telephone" name="telephone" 联系 电话 | 
text id="qq name=" qq qq | 
button id="regbtn" disabled="disabled" “注册 ”按钮 | 
button id="morebtn" 显示 /隐藏 全 部 注册 信息 | 
button id="logbtn" 返回 登录 页 面 | 
div id="imgdiv" style=" visibility: hidden:" 等 待 进度 条 | 
当 用 户 向 注册 页 面 中 的 表单 元 素 输入 信息 时 ， 将 会 触发 注册 信息 验证 脚本 registerjs。 | 
下 面 逐 行 讲解 registerjs 脚本 页 。 | 
在 registerjs 中 ,首先 写 了 一 个 通用 函数 ,来 避免 重复 输入 document.getElementById(…")。 | 


} | 
首先 将 焦点 定位 到 用 户 名 文本 框 ， 方 便 用 户 操 作 。 接 下 来 声明 了 5 个 变量 ， 这 5 个 变 

量 代表 了 5 项 要 检测 的 数据 。 当 检测 数据 为 “合格 ”时 ， 代 表 它 的 变量 将 等 于 yes。 代 码 | 
如 下 : | 


了 tp 各 重要 志 煤 种 


S(regname').focus(): // 焦 点 定位 到 用 户 名 文本 框 

// 声 明 5 个 变量 ， 表 示 要 检测 的 5 项 数据 

Var cnamel,cname2.cpwdl,.cpwd2,cemail; 

会 内 chkreg0 函 数 是 每 一 次 触发 键盘 事件 后 都 要 调用 的 。 该 函数 判断 5 个 变量 的 值 ， 只 有 
和 当 所 有 变量 都 等 于 yes 时 ，“ 注 册 ” 按 钮 才 会 被 激活 。 代 码 如 下 


Note 01 function chkregO{ 
| 02 这 (cnamel 一 "yes) && (cname2 一 'yes) && (cpwdl 一 'yes') && (cpwd2 一 'yes') 
| && (cemail 一 yes)){ 
| 03 SCregbtn).disabled = false; 
| 04 jelsef 
| 05 S$('regbtn').disabled = true; 
| 06 } 
07 } 
| 接 下 来 是 验证 用 户 名 。 当 用 户 输入 注册 名 称 时 ， 该 函数 会 把 用 户 的 每 次 输入 都 作 一 下 
| 正则 判断 ， 并 根据 结果 设置 不 同 的 cnamel 值 。 代 码 如 下 : 
| 01 /验证 用 户 名 
| 02 S(‘regname').onkeyup = function |O{ 
| 03 /获取 注册 名 称 
| 04 name = $(Tregname').value: 
| 05 cname2 ="; 
| 06 iftname.match(/^[a-zA-Z ]*/) = "){ 
| 07 S$(mamediv').innerHTML ='<font color=red> 必 须 以 字母 或 下 划 线 开头 </font>'; 
| 08 cnamel ="; 
| 09 jelse if(name.length <= 3){ 
| 10 $(namediv') innerHTML ='<font color=red> 注 册 名 称 必须 大 于 3 位 </font>'; 
| 11 cnamel ="; 
| 了 2 jelse{ 
| 13 $('namediv').innerHTML = '<font color=green> 注 册 名 称 符合 标准 </font>'; 
| 14 cnamel = 'yes'; 
| 15 } 
| 16 chkreg0: 
| 区 } 
| 当 用 户 名 文本 框 失去 焦点 时 ， 即 用 户 输入 完毕 转 到 页 面 中 其 他 元 素 的 时 候 ， 将 检测 用 
| 户 名 是 否 重 复 。 用 户 名 判断 采用 了 Ajax 技术 ， 调 用 chkname php 页 ， 并 根据 chkname .php 
| 的 返回 值 在 div 标签 中 显示 判断 结果 。chkname php 页 的 代码 将 稍 后 给 出 。 
| 01 // 当 用 户 名 文本 框 失去 焦点 时 ， 系 统 调用 处 理 页 查看 是 否 存 在 该 用 户 
| 02 S$('regname').onblur = function0{ 
| 03 name = $(regname).value: 1/ 获取 注册 用 户 名 
| 04 // 当 用 户 名 的 格式 输入 合格 后 ， 才 进行 这 一 步 操作 
| 05 if(cnamel 一 'yes){ 
| 06 xmlhttp.open('get,chkname php?name='+name,true); 。“// 创 建新 请 求 
| 07 xmlhttp.onreadystatechange = functionO{ 
| 08 if(xmilhttp readyState — 4){ 
09 if(xmlhttp.status == 200){ 


。10* 
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10 var msg = xmlhttp.responseText:; /获取 响应 页 内 容 
11 iftmsg — "1){ | 

12 S$(‘namediv').innerHTML="<font color=green> 恭 喜 您 ， 该 用 户 名 可 | 

以 使 用 !</font>"; | 
f 


13 cname2 = 'yes'; | 8 

14 jelseifmsg — 2){ | 食 - 

15 SCnamediv):i ="<font color=red> 用 户 名 被 占 | 
a (‘namediv').inner] <font color=red> 用 被 Note 


16 cname2 ="; | 
地 }else{ | 
18 S$(‘namediv').innerHTML="<font color=red>"+msg+"</font>"; 

19 cname2="; | 
20 | 
2 } | 
22 } 
23 chkreg0); /检查 是 否 要 激活 “注册 ”按钮 
24 

25 xmlhttp.send(null); 


2 } 
接 下 来 是 验证 密码 。 验 证 密码 时 ， 除 了 可 以 限制 密码 的 长 度 外 ， 还 可 以 判断 密码 的 强 
度 。 例 如 下 述 代 码 中 ， 长 度 6 一 12 之 间 的 密码 都 是 弱 ， 如 果 长 度 大 于 12， 但 密码 单一 ， 都 
是 数字 或 文字 ， 那 么 密码 强度 就 为 中 ， 否 则 强度 就 为 高 。 代 码 如 下 : 


01 /验证 密码 

02 S$(‘regpwd1'").onkeyup = functionO{ 

03 pwd = $(regpwd1').value:; 

04 pwd2 = $('regpwd2').value; 

05 if(pwd.length < 6){ 

06 $(pwddiv1").innerHTML = '<font color=red> 密 码 长 度 最 少 需 要 6 位 </font>'; 
07 cpwdl ="; 

08 }else if(pwd.length >= 6 && pwd.length < 12){ 

09 $(pwddiv1').innerHTML ='<font color=green> 密 码 符合 要 求 。 密码 强度 : 弱 </font>'; 
10 cpwdl = 'yes'; | 
11 }else if((pwd.match(/^[0-9]*$/)!=nulD) || (pwd.match(/^[a-zA-2Z]*$/) != null )){ | 
I S$(pwddiv1)innerHTML ='<font color=green> 密 码 符合 要 求 。 密码 强度 : 中 </font>; | 
13 cpwdl = 'yes'; | 
14 yelse{ | 
15 S$(pwddivl) innerHTML='<fontcolor-ereen> 密 码 符合 要 求 。 密码 强度 : 高 </font>;， 
16 cpwdl ='yes'; | 
17 } | 
18 if(pwd2 !="&&pwd!=pwd2){ | 
19 $(‘pwddiv2").innerHTML = '<font color=red> 两 次 密码 不 一 致 !</font>'; | 
20 cpwd2 ="; | 
ol }else if(pwd2 !=" && pwd — pwd2){ | 
22 $(‘pwddiv2').innerHTML = '<font color=green> 密 码 输 入 正确 </font>'; | 
23 cpwd2 = 'yes'; | 
24 FE | 
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， 料 ”按钮 。 代 码 如 下 : 


码 应 


pp 凑 齐 村 半 精 角 


chkreg(); 


二 次 密码 的 验证 比较 简单 ， 只 要 第 一 、 第 二 次 密码 相等 就 可 以 。 
// 验 证 二 次 密码 


} 


S$('regpwd2").onkeyup = functionO{ 

pwdl = $(‘regpwd1").value: 

pwd2 = $(‘regpwd2").value: 

过 pwdl != pwd2){ 
SCpwddiv2).innerHTML = '<font color=red> 两 次 密码 不 一 致 !</font>'; 
cpwd2="; 

}else{ 
$('pwddiv2").innerHTML = '<font color=green> 密 码 输入 正确 </font>'; 
cpwd2 ='yes'; 
chkreg(); 


| 接 下 来 的 E-mail 验证 主要 是 对 E-mail 的 格式 进行 检查 。 输入 的 字 串 中 , 必须 包含 “@” 
”和 “.” 同 时， 这 两 个 符号 的 位 置 既 不 能 在 首 和 尾 ， 也 不 能 连 在 一 起 。 代 码 如 下 : 

| // 验 证 E-mail 

S$(‘email).onkeyup = functionO{ 


01 
02 


} 


emailreg = /N\w+([-+.]\w+)*@\Ww+([-.]\w+)*\\w+([-.] w+)*®/; 

S$(‘email").value.match(emailreg); 

if($('email’).value.match(emailreg) 一 nul){ 
SCemaildiv).innerHTML = '<font color=red> 错 误 的 email 格式 </font>'; 
cemail 一" 

jelse{ 
SCemaildiv).innerHTML = '<font color=green> 输 入 正确 </font>'; 
cemail = 'yes'; 

}; 

chkreg(); 


上 面 的 部 分 是 对 必 填 项 的 检查 。 如 果 用 户 希 望 填写 更 详细 的 资料 ， 可 以 单 击 “ 详 细 资 


01 
02 
03 
04 
05 
06 
07 
08 


// 显 示 / 隐 藏 详细 信息 
S$(‘'morebtn'").onclick = functionO{ 


} 


if($('morediv').style.display 一 "){ 
S$('morediv').style.display = 'none': 
jelse{ 
S$('morediv').style.display = "; 
} 


所 有 信息 添加 完毕 并 通过 验证 后 ， 就 可 以 注册 了 。 用 户 单 击 “ 注 册 ” 按 钮 时 ， 下 述 代 


Ajax 技术 将 获取 的 用 户 信息 以 URL 的 形式 传 给 register_chk.php 页 , 并 根据 返回 结 
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果 作 出 相应 的 处 理 。 代 码 如 下 : 


1.3.3 ”检测 用 户 名 是 否 被 占用 


检测 用 户 名 是 否 被 占用 需要 用 到 chkname.php 文件 ， 其 作用 是 检查 用 户 名 是 否 重复 ， 
并 且 返 回 判断 结果 。 代 码 如 下 : 

01 <?php | 
02 ”// 载 入 数据 库 类 | 
03 include_once "conn/conn.php"; 
04 /根据 name 查询 tb_member 表 
05 $sql = "select * from tb_member where name=".$_ GET[name']."": | 
06 /获取 查询 结果 的 记录 数 | 
07 $num = $conne->getRowsNum($sql): 


01 /正式 注册 | 

02 $(Cregbtn).onclick = functionO{ | 

03 S$(imgdiv').style.visibility = 'visible'; | 僚 站 

04 Url = ‘register chk.php?name='+: le").valuet'&pwd='+$('regpwd1').valuet+'&email | 一 一 
='+9(0email).value; 

05 Url += '&question=" +$(question).value+'&answer='+9(0answer).value; Note 

06 Url +1= '&realname=' +$('realname').value+'&birthday="+$('birthday").value; | 

07 url += '&telephone='+$('telephone').valuet+'&qq='+$('qq'").value; | 

08 alert(ur); | 

09 return false; | 

10 xmlhttp.open('get',url,true); | 

11 xmlhttp.onreadystatechange = functionO{ | 

12 if(xmlhttp.readyState 一 4){ | 

13 if(xmlhttp.status == 200){ | 

14 msg = xmlhttp.response Text: | 

15 if(msg =—'1){ | 

16 alert( 注 册 成 功 ， 请 到 您 的 邮箱 中 获取 激活 码 ! ); | 

17 location='main php'; | 

18 Yelse if(msg =="-1){ | 

19 alert( 您 的 服务 器 不 支持 POP3， 请 仔细 检查 !! 小 | 

20 yelse{ | 

ol alert(msg); | 

22 } | 

23 } | 

24 S$(‘imgdiv'").style.visibility = ‘hidden'; | 

65 3 | 

26 } | 

2 xmlhttp.send(null): | 

28 } | 


08 ifl$num— D{ // 如 果 有 记录 | 
09 echo 2'; | 
10 J}elseif(Snum— 0){ // 如 果 没 有 记录 | 
11 echo '1'; | 
jb }else{ | 
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13 echo $conne->msg_ error(); // 否 则 出 错 
14 } 
15 ?> 
| 1.3.4 “注册 信息 处 理 
在 注册 信息 处 理 页 (register_chk.php) 中 ， 将 JavaScript 传递 过 来 的 值 连 成 insert 语句 


| 并 执行 。 如 果 添 加 成 功 ， 则 使 用 邮件 函数 发 送 一 封 激活 邮件 ， 并 弹出 提示 对 话 框 ， 如 图 1.8 


图 1.8 注册 成 功 提示 对 话 框 


| 在 处 理 代码 中 包含 两 种 邮件 的 发 送 方法 ,一 种 是 使 用 局 域 网 ， 向 指定 SMTP 服务 器 中 
“的 邮箱 (这 里 的 cym3100@163.com 邮箱 是 在 SMTP 服务 器 中 注册 的 测试 邮箱 ) 中 发 送 邮 
， 件 ， 另 一 种 是 使 用 互联 网 ， 通 过 指定 的 邮箱 (mrsoft8888@sohu.com 是 在 搜狐 网 中 注册 的 


| 邮箱， 用 户 名 为 mrsoft8888， 密 码 为 mrsoft8888) 向 用 户 提供 的 有 效 邮 箱 中 发 送 邮件 。 处 
， 理 页 的 代码 如 下 : 
| 01 <?php 
| 02 include_once 'conn/conn.php'; 
| 03 require_once 'Zend/Mailphp': /调用 发 送 邮件 的 文件 
| 04 Tequire_once 'Zend/Mail/Transport/Smtp.php'; // 调 用 SMTP 验证 文件 
05 S$reback = '0'; 
| 06 $url = 'http://.$_SERVER['SERVER_NAME'].dimame($_SERVERI['SCRIPT_ NAME']). 
| /activation.php'; 
| 07 Surl .= ?name=".trim($_GET['name']).'&pwd="'.md5(trim($_GET['pwd'])); 
| 08 /发 送 激活 邮件 
| 09 $subject=" 激 活 码 的 获取 "; 
| 10 $mailbody=' 注 册 成 功 。 您 的 激活 码 是 : 
| 11 "<a hre 全 "Surl target=" blank">'Surl'</a><br>'…" 请 点 击 该 地 址 ， 激 活 您 的 用 户 ! '; 
| 12 // 定 义 邮 件 内 容 
| 13 Senvelope="mrsoft8888(@sohu.com"; // 定 义 登录 使 用 的 邮箱 
| 14 ” 族 。”SMTP 测试 版 发 送 邮 件 方法 ， 使 用 SMTP 作为 服务 器 
| 15 S$tr = new Zend Mail Transport Smtp('192.168.1.247"): 
| 16 Smail = new Zend_ Mail0: 
| 1 $mail->addTo('cym3100@163.com', 获 取 用 户 注册 激活 码 '); 
| 18 $mail->setFrom('cym3100@163.com',' 明 日 科技 典型 模块 程序 测试 邮箱 ， 巷 喜 您 用 户 
”注册 成 功 ! ”); 
| 19 Smail->setSubject(' 获 取 注册 用 户 的 激活 码 '); 
20 Smail->setBodyHtml($mailbody): 
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5 Smail->send($tr); | 
22 4 | 
23 ”网 络 版 发 送 邮 件 方法 */ | 
24 $config = array(auth' => login', | 
25 username' => ‘mrsoft8888', | 到 六 
26 ‘password' => ' mrsoft8888"); // 定 义 SMTP 的 验证 参数 | 痊 一 
27 $transport = new Zend Mail Transport Smtp('smtp.sohu.com', $config); | 
// 实 例 化 验证 的 对 象 ”JVote 
28 Smail = new Zend Mail(GBK'): // 实 例 化 发 送 邮 件 对 象 | 
29 Smail->setBodyHtml($mailbody); // 发 送 邮件 主体 | 
30 。 $mail->setFrom(genvelope,' 明 日 科技 典型 模块 程序 测试 邮箱 .恭喜 您 用 户 注册 成 功 ! "); | 
/定义 邮件 发 送 使 用 的 邮箱 | 
3 $mail->addTo($_GET[email], ' 获 取 用 户 注册 激活 码 ); /定义 邮件 的 接收 邮箱 | 
32 Smail->setSubject(' 获 取 注 册 用 户 的 激活 码 "); // 定 义 邮件 主题 | 
33 Smail->send($transport); // 执 行 发 送 操作 | 
34 。 /# ”网 络 版 发 送 邮 件 方法 */ | 
35 $sql = "insert into tb_ member(name,password,question,answer,email,realname,birthday, | 
telephone,qq) | 


36 values(".trim($_GET[name']).",".mds(trim($_ GET[pwd]))."".$_GET['question']."".3 
GET['answer'].", 

37 "S$ GET['email].",".$ GET['realname'].",".$ GET['birthday'].",".$_GET["telephone'].", 
".$_GET['gq].")"; 


38 $num = $conne->uidRst($sq]); 
Es if($num 一 1){ 

40 Sreback = "1"; 

41 } 

42 echo S$reback:; 

43 ?> 


1.3.5 ”注册 用 户 激活 


在 注册 成 功 后 ， 用 户 需要 登录 自己 填写 的 邮箱 ， 浏 览 注 册 时 发 送 的 邮件 ， 并 且 单 击 超 
链接 ， 链 接 到 activationphp 文件 ， 根 据 超 链接 传递 的 用 户 名 和 密码 完成 对 指定 用 户 表 中 | 
active 字段 值 的 更 新 ， 即 完成 注册 用 户 的 激活 操作 。 | 


登录 mrsoft8888@sohu.com 邮箱 ， 将 查看 到 如 图 1.9 所 示 的 邮件 内 容 。 


和 查看 闻 件 
时 间 : 2011-04-29 13:1:41 ! 
发 件 人 : spBnrbccd com 
收 件 人 : spBnrbecd com 
主题 : 激活 码 的 获取 | 
注册 成 功 。 您 的 激活 码 是 : http://192. 168. 1. 149/model/login/ activation php? 1 


name=mrhpwd=fdb390e945559eT44T5ed8c8bbb46ca5 ! 
请 点 击 该 地 址 ， 激 活 您 的 用 户 ? 


图 1.9 激活 邮件 的 运行 结果 
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Note 


进行 激活 后 ， 用 户 才 可 以 使 用 。 这 里 向 读者 介绍 一 种 不 需要 激活 的 用 户 注册 方式 ， 同 样 以 
| 上 面 注册 提交 的 数据 为 基础 ， 只 需 修改 用 户 注册 提交 数据 的 处 理 方式 ， 就 可 以 实现 免 激 活 
| 的 用 户 注册 功能 。 该 操作 在 register_check.php 文件 中 完成 。 其 关键 代码 如 下 : 


Ga pp 前 杭 决 精 角 
2 


| 当 用 户 进入 激活 页 (activation.php) 后， 激活 页 将 根据 name 字段 和 pwd 字段 来 查找 
| 匹配 的 注册 用 户 ， 如 果 找 到 将 进行 更 新 操作 。 激 活 成 功 后 ， 将 用 户 名 保存 到 Session 中 
， 并 跳 转 到 相应 的 页 面 。 代 码 如 下 : 

| 01 < ?php 

02 session start(); 


03 header('Content-Type:text/html:;charset=gb2312"); 
04 include_once("conn/conn.php"); 


05 if(lempty($ GET[‘name']) && !is null($ GET['name']){ // 激 活 注册 用 户 
06 $num=$conne->getRowsNum("select * from tb _ member 

07 where name=".$_GET[name']." and password =".$_ GET['pwd'].""); 

08 if (Snum>0){ 

09 Supnum=$conne->uidRst("update tb_ member set active = 1 

10 where name=".$ GET['name']." and password =".$_ GET['pwd']." 

11 if($upnum > 0){ 

12 $_SESSION[name'] = $_GET[mame']: 

13 echo "<scripf>alert( 用 户 激活 成 功 ! ");windowJlocation hre 人 = main.php'; </script>"; 
14 }else{ 

15 echo "<script>alert( 您 已 经 激活 ! ');window.location hre 人 main php': </script>"; 
16 } 

17 jelse{ 

18 echo "<script>alert( 用 户 激活 失败 ! '):window.location href='registerphp'; </script>"; 
19 } 

2000 

21 ?> 


,1.3.6 免 激活 用 户 注册 


上 面 介绍 的 用 户 注册 功能 中 ， 需 要 将 激活 信息 以 邮件 的 形式 发 送 到 用 户 指定 的 邮箱 中 


01 <?php 

02 include_once 'conn/conn.php'; // 包 含 数据 库 连接 操作 文件 

03 Sreback = '0'; // 定 义 初始 变量 
| 04 $sql = "insert into tb_member(name,password,question,answer,email,realname,birthday, 
| telephone,qq,active) 
| 05 values(".trim($ GET['name']).",".md5(trim($ GET['pwd'])).".".$_GET['guestion].",".$ 
GET['answer'].", 
| 06 "S$ GET[email].",".$ GET['realname’].",".$ GET['birthday'].",".$_GET['telephone'].",' 
| "$8_GET['qq]. wz 
| 07 $num = $conne->uidRst($sql); // 执 行 添加 语句 

08 if($num — D){ 

09 Sreback = "1'; 

10 } 

11 echo $reback: // 输 出 变量 值 

12 全 
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说 明 : 


在 本 模块 中 要 使 用 免 激 活 模式 来 注册 用 户 , 首先 应 该 在 js\registerjs 文件 中 修改 注册 数 
据 处 理 URL 路 径 ， 将 注册 数据 处 理 文件 ( register_chk.php ) 修改 为 register check.php。 代 


码 如 下 : 


01 
02 
03 


ul 


S$('regbtn').onclick = functionO{ 


S$(imgdiv'").style.visibility = 'visible'; 


= ‘Tegister chk.php?name='+$('regname').value+'&pwd='+$('regpwd1').valuet+ 


'&email z='+$('email").value; 


如 


一 来 , 在 执行 用 户 注册 数据 的 处 理 时 , 就 会 调用 上 面 重新 编写 的 register_check.php 


文件 ，| 


1.4.1 


} 


Url += '&question=" +$('question').valuet+'&answer='+$("answer').value; 
Url += '&realname=' +$('realname').value+'&birthday="+$('birthday").value; 
Url += '&telephone='+$('telephone').valuet+'&qq='+$('qq'").value; 
xmlhttp.open('get',url,true): 
xmlhttp.onreadystatechange = function|{ 
if(xmlhttp.readyState 一 4){ 
if(xmlhttp.status == 200){ 
msg = xmlhttp.responseText: 
if(msg =—'1){ 
alert( 注 册 成 功 ， 请 到 您 的 邮箱 中 获取 激活 码 ! ，); 
location="index.php'; 
jelse if(msg =— -1){ 
alert(' 您 的 服务 器 不 支持 Zend_mail， 或 者 邮箱 填写 错误 。 


}else{ 
alert(msg); 
b 
S$(imgdiv').style.visibility = "hidden'; 
} 
} 
} 
xmlhttp.send(null); 


户 注册 成 功 后 可 以 直接 登录 ， 而 不 必 再 进行 激活 操作 。 


1.4 用 户 登 录 


用 户 登录 功能 概述 


用 户 登录 功能 即 对 用 户 输入 的 账号 、 密 码 以 及 验证 码 进行 验证 。 验 证 内 容 包括 : 信息 是 
否 为 空 、 验 证 码 是 否 错误 、 用 户 计算 机 是 否 已 经 被 锁定 、 用 户 是 否 存在 、 用 户 是 否 激活 、 用 


户 名 和 密码 是 否 匹 配 、 用 户 登 录 次 数 是 否 达到 上 限 等 。 登 录 功 能 的 运行 结果 如 图 1.10 所 示 。 
登录 功能 包含 3 个 核心 文件 ， 分 别 是 loginphp〔 用 户 登录 页 )、loginjs〈 登 录 信息 验 
证 脚本 ) 和 login_chk.php〈 登 录 处 理 页 )。 
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免费 下 雪 。。 在 线 订 胃 


加 果 作 流下 注册 。 请 您 证 骨 后 再 号 录 。 如 果 容 已 径 广 骨 ,但 地 证 了 袜 码 , 您 可 以 通过 
后 衬 三 村 并 的 窑 码 找 回 


本 录 和 名称: nr 


Ew 
克 本 |] 59 寿 直 


验证 码 : 
EE Esa EE 


图 1.10 单 击 “ 登 录 ” 按 钮 后 的 运行 结果 
1.4.2 ”登录 页 面 设计 
| 1 用 户 登 录 页 (以 下 简称 登录 页 ) 


相对 于 注册 页 面 来 说 ， 登 录 页 (login.php) 中 的 表单 元 素 要 少 很 多 ， 也 没有 那么 复杂 。 
| 登录 页 中 使 用 的 元 素 如 表 1.2 所 示 。 
| 表 1.2 登录 页 中 的 元 素 


元 素 名 称 元 素 属性 值 说 明 
script "js/xmlhttprequest.js" 载 入 xmlhttprequest 对 象 脚 本 
Script "js/login.js" 载 入 登录 信息 验证 脚本 
text id="lgname" name="lgname" 注册 名 称 
password id="lgpwd" name="lgpwd" 注册 密码 
text id="lgchk" name="lgchk" 验证 码 
img id='chkid' sre="" width="55" height="18" 图 形 验 证 码 
| a id="changea" 更 换 验 证 码 超 链接 
| hidden id = "chknm" name = "chknm" value="" 隐藏 域 ， 保 存 验证 码 
| button id="lgbtn" “登录 ”按钮 
| button id="reg" 跳 转 到 注册 页 
button id="found" 找 回 密码 


2. 登录 信息 验证 脚本 (loginjs ) 


在 登录 页 login php 中 ， 载 入 了 loginjs 脚本 文件 ， 首 先 通过 该 脚本 文件 完成 对 登录 用 
| 户 的 所 有 验证 操作 。 代 码 如 下 : 
| 01 function $(id){ 


| 02 return document.getElementById(id): 
| 0 

| 04 window.onload = function0){ 

| 05 se 

| 06 } 
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接 下 来 的 函数 功能 是 焦点 自动 下 移 。 当 焦点 移动 到 最 后 一 个 文本 框 即 验 证 码 文本 框 | 
后 ， 再 次 按 下 Enter 键 ， 将 调用 chklg0 函 数 对 用 户 输入 的 信息 进行 验证 。 实 现 原理 将 在 后 | 
面 的 “技术 提炼 ”中 进行 详细 讲解 。 代 码 如 下 : 


ol SClgname) focusO; /光标 定位 到 用 户 名 文本 框 加 会 内 
02 S$(lgname').onkeydown = functionO{ // 处 理 用 户 名 文本 库 的 键盘 按 下 事件 Shu 
03 ieventkeyCode 一 13){ // 当 按 下 Enter 键 时 ， 光 标 自动 下 移 ‘Note 
04 S(lgpwd').select(); /密码 文本 框 获取 焦点 | 

05 } | 

06 } | 

07 S$(lgpwd').onkeydown = functionO{ | 

08 这 eventkeyCode 一 13){ /处 理 密码 文本 框 的 鼠标 按 下 事件 | 

09 $(lgchk').selectO; // 验 证 码 文本 框 获得 焦点 | 

10 } | 

11 y | 

12 S$('lgchk").onkeydown = function0{ /处 理 验证 码 文本 框 的 鼠标 按 下 事件 

13 这 eventkeyCode 一 13){ 

14 chklg0; /调用 chklg0) 函 数 

15 } 

16 } 

17 $(lgbtn").onclick = chklg: 


然后 进行 最 基本 的 检测 ， 包 括 用 户 名 是 否 合法 、 信 息 不 能 为 空 、 验 证 码 是 否 输入 错误 
等 。 代 码 如 下 : 
01 function chklgO{ 


02 旋 判 断 用 户 名 是 否 合法 ， 用 户 名 第 一 位 必须 是 英文 字母 ， 不 允许 有 特殊 字符 */ 
03 if($(lgname').value.match(/^[a-zA-Z_ Mw*$/) =— nul){ 

04 alert(' 请 输入 合法 名 称 '); 

05 S$('lgname').select(); | 
06 return false: | 
07 } | 
08 必用 记 才 十 百 为 空 尖 六 | 
09 if($(lgname').value — "){ | 
10 alert(' 请 输入 用 户 名 由; | 
11 $(lgname').focusO; | 
12 retum false; | 
13 } | 
14 /# 密码 是 否 为 空 */ | 
15 ifSClgpwd) value 一 9){ | 
16 alert(' 请 输入 密码 !"); | 
17 S$(lgpwd').focus(); | 
18 Teturn false: | 
19 } | 
20 /# ”验证 码 是 否 为 空 */ | 
2all if($(lgchk'").value — "){ | 
2 alert(' 请 输入 验证 码 ): | 
23 S$('lgchk').select(); | 
24 Teturn false: | 
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25 

26 人 # ”验证 码 输 入 是 否 正确 */ 

2 if($('lgchk'’).value != $('‘chknm").value){ 
28 alert(' 验 证 码 输入 错误 '); 

29 S$('lgchk').selectO; 

30 Teturn false; 

3 站 


| 接 下 来 ， 判 断 客户 端的 Cookie 值 是 否 大 于 3， 如 果 大 于 3， 说 明 / 
| 将 停止 继续 执行 。 代 码 如 下 ; 


户 是 非法 操作 的 ， 


| 01 count = document.cookie.split(;")[0]; /获取 Cookie 信息 
| 02 if(count.split(=")[1] >= 3){ 

03 alert( 因 为 您 的 非法 操作 ， 您 将 无 法 再 执行 登录 操作 '); 

04 return false; 

05 } 


最 后 ， 将 用 户 输入 的 信息 传递 给 login_chk .php 页 进行 处 理 。 代 码 如 下 : 


| 01 ul = 'login chk.php?act='+(Math.random())+'&name='+$('lgname').valuet'&pwd 
| =+$(lgpwd').value; 

02 xmlhttp.open('get',url,true); 

03 xmlhttp.onreadystatechange = functionO{ 

04 if(xmlhttp.readystate 一 4){ 

05 if(xmlhttp.status 一 200){ 

06 /获取 login_chk.php 页 的 返回 值 

07 msg = xmlhttp.response Text: 

08 if(msg 一 '0){ 

09 alert(' 您 还 没有 激活 ， 请 先 登 录 邮 箱 进行 激活 操作 。); 

10 jelse if(msg == "1"){ 

11 alert( 用 户 名 或 密码 输入 错误 ， 您 还 有 2 次 机 会 '); 

12 S$(lgpwd').select(): 

13 }else if(msg == '2"){ 

14 alert( 用 户 名 或 密码 输入 错误 ， 您 还 有 1 次 机 会 '); 

15 S$(lgpwd').select(): 

16 } else ifmsg 一 '4){ 

I alert( 用 户 名 输入 错误 "); 

18 S$(lgname').select(): 

19 } else iftmsg —'-1){ 

20 alert(' 登 录 成 功 "); 

| location = "main.php'; 

22 }else{ 

23 alert(msg); 

24 } 

25 } 

26 

27 } 

28 xmlhttp.sendCnull): 

29 } 
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1.4.3 ”登录 处 理 页 
在 登录 处 理 页 (login_chkphp) 中 ， 对 表单 中 提交 的 用 户 名 、 密 码 进行 验证 ， 包 括 用 | 


户 名 是 否 存在 、 用 户 名 是 否 被 激活 、 用 户 登录 次 数 、 密 码 是 下 正确 等 ,最终 返回 验证 结果 。 oy 
代码 如 下 ; | 使 - 


01 <?php Note 


02 session start(); /开启 Session 支持 | 
03 header('Content-Type:text/html:charset=gb2312"; /设置 头 信息 | 
04 include_once 'conn/conn.php'; // 载 入 数据 库 类 | 
05 $name =$_ GET[mame']: | 
06 Spwd=$_GET['pwd]; /获取 名 称 和 密码 | 
07 这 !empty($name) and !empty($pwd){ | 
08 // 得 到 当前 数据 表 中 的 count 字段 和 active 字段 | 
09 $sql = "select name.count,active from tb_member where name = ".$name."™"; | 
10 Sactive = $conne->getFields($sqlL2): | 
11 $count = $conne->getFields($sql,1): | 
12 S$conne->close_Ist(); | 
13 /判断 用 户 是 否 存在 | 
14 if($active 一 "){ | 
15 iflis_null($ COOKIE['count']) or $ COOKIE['count] == 0){ | 
16 setcookie('count',1); | 
17 }else{ | 
18 setcookie('count',$ COOKIE['count]+1): | 
19 } | 
20 Sreback = 4; | 
21 }else if($active 一 0){ // 判 断 用 户 是 否 被 激活 | 
22 Sreback = "0'; | 
23 }else if($count >= 3){ | 
24 Sreback = "3'; 1/ 判断 用 户 的 登录 次 数 | 
25 jelse{ | 
26 $sql .= " and password = ".md5($pwd)."": | 
2 $num = $conne->getRowsNum($sq]): | 
28 if($num — 0 or $num — "){ // 如 果 用 户 密码 错误 , 登录 次 数 加 1 | 
29 $num = $conne->uidRst("update tb_ member set count = ".($count+1)." where name = | 
Sname wm) | 
30 Sreback = ($count+ 1); | 
31 Yelse{ 1/ 登录 成 功 ， 清 空 count 字段 值 | 
32 if(Scount (= 0){ | 
EE) Snum = $conne->uidRst("update tb_member set count = 0 where | 
name = ".$name.""); | 
34 } | 
35 // 设 置 cookie | 
36 iffisset($_COOKIE['count]) and S_COOKIE['count] != 0){ | 
37 setcookie('count',0); | 
38 } | 
39 $_SESSION[mame] = $name: | 
40 Sreback = -1'; | 
41 } | 
42 } | 
43 } | 
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44 echo S$reback: 


a | 1.4.4 ”生成 及 刷新 验证 码 
A 


验证 码 生成 应 用 的 是 GD2 函数 库 中 的 函数 ， 操 作 存储 于 valcode php 文件 中 ， 其 完整 
芝 CLX 王 代码 参见 1.6.3 节 的 内 容 ， 这 里 不 再 效 述 。 
。。。 当 页 面 第 一 次 被 载 入 时 ， 将 执行 loginjs 中 的 showval0 函 数 来 生成 验证 码 ， 并 且 调 用 
| valcode.php 文件 以 图 像 形 式 输出 验证 码 。 因 为 验证 码 中 带 有 干扰 串 ， 如 果 用 户 看 不 清 ， 可 
”以 单 击 首页 中 的 “看 不 清 ” 超 链接 来 再 次 执行 shwoval0 函 数 ， 即 刷新 验证 码 。loginjs 脚 


| 本 中 showval0 函 数 的 代码 如 下 : 
| 01 showvalO; 1/ 调用 showval0 函 数 生成 图 文 验证 码 
02 S$(‘changea').onclick= showval; // 更 换 验 证 码 
03 /验证 码 生 成 函数 
04 function showvalO{ 
05 Dm = 
06 for(i=0;i<4:it+){ 
07 人 // 生 成 随机 数 
08 
09 S$Cchkid).src='valcode php?num='+num:// 显 示 图 文 验证 码 
10 S$('chknm').value = num:; // 将 验证 码 保存 到 chnm 隐藏 域 
11 } 
1.5 找 回 密码 


1.5.1 找 回 密码 功能 概述 


| 找 回 密码 功能 用 到 的 技术 也 是 邮箱 技术 。 用 户 输 入 正确 的 账号 、 密 保 问题 、 密 保 答案 
| 后， 系统 将 随机 生成 的 一 组 数字 发 送 到 用 户 的 邮箱 中 ， 用 户 可 以 通过 邮箱 来 得 到 新 密码 。 
| 找 回 密码 功能 的 运行 结果 如 图 1.11 所 示 。 


找 回 密码 - 
[®@) hapy/1921681.59/soFUoUfoundphp lil| 
室友 找 回 
找 加 账号 : pr 


密 保 问题 围 日 科技 
密 保 答案 ， 图 书 开发 


看 Internet | 保护 模式 华 组 所 100% ~ | 
人 ee 


图 1.11 找 回 密码 功能 的 运行 结果 


有 Cp 


第 ] 章 ”注册 登录 模块 Yo 
4 
1.5.2 ” 找 回 密码 功能 的 实现 过 程 


找 回 密 码 操 作 从 单 击 登录 页 面 中 的 “ 找 回 密码 ”按钮 开始 ， 在 弹出 的 found.php 文件 
中 创建 一 个 表单 ， 提 交 找 回 账号 、 密 保 问题 和 密 保 答 案 ， 并 且 载 入 xmlhttprequestjs 和 
found.js 脚本 文件 。found .php 的 关键 代码 如 下 : 


01 <link rel="stylesheet" hre 合 "css/style.css" /> 

02 <script language="javascript" sre="js/found.js"></script> | 
03 <script language="javascript" src="js/xmlhttprequest.js"></script> | 
04 <div id="fdbgdiv" > | 
05 <div id="top">&nbsp;>> 密 码 找 回 </div> | 


06 <div id="foundnamediv"> 找 回 账号 : | 
07 <input id="foundname" type="text" style=" width: 100px; height:1Spx; border:lpx #000000 '! 
solid;" /></div> 
08 <div id="foundnamediv"> 密 保 问 题 : | 
09 <input id="fdquestion" type="text" style=" width: 100px: height:15px; border:lpx #000000 | 
solid;" /></div> | 
10 <div id="foundnamediv"> 密 保 答案 : | 
11 <input id="fdanswer" type="text" style=" width: 100px: height:15px; border:lpx #000000 | 
solid;" /></div> | 
12 <div id="foundnamediv" align="center"><button id="step1"></button></div> | 
13 </div> | 


在 foundjjs 文件 中 , 调用 found_chk.php 文件 , 完成 密码 的 找 回 操作 。 其 关键 代码 如 下 : 
01 function SGid){ 


02 Teturn document.getElementById(id); 
03 } 
04 window.onload = function0){ 
05 $('foundname').focus(); | 
06 S$('step1").onclick = functionO{ | 
07 if($(foundname).value != " && S$(fdquestion)value {= " && | 
S$('fdanswer').value !="){ | 
08 xmlhttp.open('get','found_chk.php?foundname="+$('foundname'). | 
09 valuet'&question="+$('fdquestion'").valuet'&answer="+$('fdanswer').value,true); | 
10 xmlhttp.onreadystatechange = function(O{ | 
11 这 xmlhttp readyState == 4 && xmlhttp.status == 200){ | 
12 msg = xmlhttp.responseText: | 
13 ifmsg 一 '10{ | 
14 alert(' 找 回 密码 成 功 ， 请 登录 注册 邮箱 !"); | 
15 window.closeO; | 
16 jelsef | 
记 alert( 填 写 信息 错误 ! ): | 
18 } | 
19 } | 
20 } | 
21 xmlhttp.send(null); | 
2 jelsef | 
孝 alert( 请 填写 完整 信息 : | 


。23 。 


24 
25 
26 
27 
28 
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S$('‘foundname'").focus(): 
return false: 


在 found_chk.php 文件 中 ， 对 用 户 提交 的 密 保 问题 和 密 保 答 案 进 行 验证 ， 如 果 正 确 则 


| 生成 新 的 密码 ， 并 且 更 新 数据 库 中 存储 的 密码 ， 同 时 将 新 密码 发 送 到 用 户 注册 时 填写 的 邮 
| 箱 中 。found_chkphp 的 关键 代码 如 下 : 


01 <?php 

02 include_once 'conn/conn.php'; 

03 require_once ‘Zend/Mail.php'; /调用 发 送 邮件 的 文件 

04 require_once 'Zend/Mail/Transport/Smtp.php'; // 调 用 SMTP 验证 文件 

05 Sreback = "0'; 

06 S$name = $_GET['foundname']; 1/ 获取 用 户 名 

07 S$question =$_GET['question']; // 获 取 密码 保护 问题 

08 Sanswer =$_GET['answer]; /获取 密码 保护 的 答案 

09 $sql = "select email from tb_member 

10 Where name = ".$name." and question = ".$question." and answer =".$answer"":; 多 定义 
查询 语句 

11 $email = $conne->getFields($sql,0); 1/ 判断 用 户 提交 的 用 户 密 保 是 
否 正 确 

12 if($email ="){ // 如 果 正 确 则 执行 如 下 操作 

13 S$md = rand(1000,time()); // 生 成 随机 数 

14 $sql = "update tb_member set password = ".md5($rnd)." 

| 15 wherename=".$name." and question=".$question."and answer=".$answer""; /更 新 
| 密码 

16 S$tmpnum = $conne->uidRst($sq]); /执行 更 新 操作 

17 if($tmpnum >= 1){ 

18 /发 送 密码 邮件 

19 $subject=" 找 回 密码 "; 

20 S$mailbody=' 密 码 找 回 成 功 。 您 账号 的 新 密码 是 .$md: 

2 Senvelope["from"]="cym3100(@163.com"; 

22 //$envelope="mrsoft8888(@sohu.com"; /网 络 版 定义 登录 使 用 的 邮箱 

23 上 ”SMTP 测试 版 发 送 邮件 方式 ， 使 用 SMTP 作为 服务 器 */ 

24 S$tr = new Zend_Mail Transport_Smtp(192.168.1.247)): 

23 Smail = new Zend_ Mail0: 

26 $mail->addTo($email,' 获 取 用 户 新 密码 "); 

27 Smail->setFrom('cym3100@163.com',' 明 日 科技 典型 模块 程序 测试 邮箱 , 修 
改 用 户 注册 密码 ! "); 

28 Smail->setSubject(Ssubjecb: 

29 Smail->setBodyHtml(Smailbody): 

30 //$Smail->send($tr); 

31 谨 网络 版 发 送 邮 件 方 法 */ 

32 /*$config = array('auth' => 'login', 

33 msemame' => 'mrsoft8888’', 

34 password' => ' mrsoft8888"); // 定 义 SMTP 的 验证 参数 

3 Stransport = new Zend Mail Transport Smtp('smtp.sohu.com', $config); /实例 
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化 验证 的 对 象 
36 Smail = new Zend Mail(GBK'); // 实 例 化 发 送 邮 件 对 象 
Sm Smail->setBodyHtml($mailbody): // 发 送 邮 件 主体 
38 $mail->setFrom($envelope, ' 明 日 科技 典型 模块 程序 测试 邮箱 , 修改 用 户 注册 密码 ! ); 
39 $mail->addTo($email, ' 获 取 用 户 新 密码 ); /定义 邮件 的 接收 邮箱 
40 Smail->setSubject($subject); // 定 义 邮件 主题 
41 Smail->send($transport); // 执 行 发 送 操作 */ 
42 ” 启 ” 网 络 版 发 送 邮 件 方法 */ 
43 if(false —$mail->send($tr) ){ 
4 Sreback ="-1'; 
45 jelse{ 
46 Sreback = "1'; 
47 } 
48 }else{ 
49 S$reback = 2'; 
s000 
51 jelse{ 
52 Sreback = $sql; 
S30 
54 echo Sreback:; 
S57 


1.6.1 防 SQL 注入 技术 


SQL 注入 , 是 指 一 些 精通 SQL 语句 的 用 户 , 通过 在 表单 或 浏览 器 地 址 栏 中 输入 SQL 语 
句 来 绕 过 系统 检验 的 一 种 技术 。 一 般 防范 SQL 注入 的 手段 是 过 滤 敏 感 字符 ， 例 如 引号 等 。 | 
相对 于 ASP 来 说 ，PHP 要 安全 得 多 , 但 这 不 代表 PHP 网 站 就 不 会 出 现 SQL 注入 的 情 | 
况 。 幸 运 的 是 ， 在 PHP 上 防 SQL 注入 ， 要 比 ASP 简单 、 方 便 ， 不 需要 编写 一 大 段 的 转换 | 
语句 ， 只 要 几 个 函数 就 可 以 。 | 


(1) intval0 函 数 


intval(0) 函 数 的 作 | 
int intval ( mixed 


(2) addslashes() 


是 返回 


函数 


addslashes() 函 数 用 


1.6 技术 提炼 


变量 的 整数 值 ， 其 语法 如 下 : 


var [, int base] ) 


符 前 加 上 反 斜 杠 A\)， 包 括 单 引号 (')、 双 引号 (")、NULL， 但 是 不 包括 “%” 和 “_”。 


其 语法 格式 如 下 : 


string addslashes ( string str ) 


此 外 ， 还 可 以 使 用 mysql_real escape_string() 来 进行 转 义 ， 效 果 和 addslashes() 函 数 是 


一 样 的 。 
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于 在 操作 数据 库 时 ， 对 其 中 的 特殊 字符 进行 自动 转 义 ， 即 在 特殊 字 


到 VAI 


1.6.2 Ajax 无 刷新 验证 技术 


Ajax (Asynchronous JavaScript and XML， 异 步 JavaScript 和 XML) 是 时 下 最 流行 的 


技术 。 它 并 不 是 新 的 技术 ， 而 是 原 有 技术 的 集合 ， 这 从 它 的 名 称 上 就 能 够 看 出 来 。 


Ajax 的 核心 技术 是 XMLHttpRequest。 通 过 XMLHttpRequest 中 的 open() 方 法 和 send() 


| 方法 ， 可 以 在 不 刷新 当前 页 面 的 情况 下 向 处 理 页 发 送 数据 ;通过 XMLHttpRequest 中 的 
| responseText 属性 和 responseXML 属性 ， 可 以 得 到 处 理 页 的 输出 结果 。 


Ajax 之 所 以 能 够 流行 ,是 因为 它 能 为 用 户 提供 良好 的 交互 性 。 这 个 特性 在 本 模块 中 被 


充分 地 体现 出 来 ， 注 册 用 户 无 须 经 过 “漫长 ”的 等 待 就 可 以 知道 输入 的 账号 是 否 可 用 ， 登 
录用 户 也 不 必 担 心 因为 输 错 登 录 信 息 而 重新 刷新 整个 页 面 。 


使 用 Ajax， 一 般 分 为 下 面 几 步 。 


(1) 首先 要 创建 XMLHttpRequest 对 象 。 不 同 的 浏览 器 ， 创 建 XMLHttpRequest 对 象 


及 使 用 的 方法 有 一 些 差别 。 这 里 只 针对 正 浏览 器 进行 创建 ， 代 码 如 下 : 


01 var xmlhttp = false; /初始 化 变量 
02 ”// 如 果 ActiveXObject 存在 ， 说 明 是 下 1.0 以 上 的 版 本 ， 和 否则 使 用 XMLHttpRequest 创建 
03 if(window.ActiveXObject) { 


04 xmlhttp = new ActiveXObject("Microsoft. XMLHTTP"): 
05 }else if(window.XMLHttpReuqest){ 
06 xmlhttp = new XMLHttpRequest(): 
07 } 
(2) 对 象 创建 成 功 后 , 就 可 以 使 用 对 象 中 的 open0 方 法 创建 新 请 求 了 。 其 语法 格式 如 下 : 
xmlhttp.open(rmethod,rurl,isAsynce); 
参数 说 明 如 下 。 


回 rmethod: 指明 请 求 的 方法 ， 如 get0 或 post()。 

回 rurl: 指明 请 求 的 页 面 。 可 以 是 绝对 地 址 ， 也 可 以 是 相对 地 址 。 

isAsync: 指明 请 求 是 否 为 异步 。 默 认为 Tue， 即 异步 。 

(3) 如 果 isAsync 等 于 True， 那 么 当 请 求 的 状态 改变 时 ， 将 调用 onreadystatechange 


属性 ， 该 属性 指定 了 一 个 回调 函数 。 其 语法 格式 如 下 ; 


xmlhttp.onreadystatechange = reabackfunc; 


| 或 者 


xmlhttp.onreadystatechange = functionO{**…} 
(4) 在 回调 函数 中 ， 首 先 需 要 判断 HTTP 的 请 求 状态 和 “HTTP 状态 码 。 这 是 通过 


”readyState 属性 和 status 属性 来 判断 的 。readyState 属性 有 5 种 状态 值 ， 常 用 的 是 4， 表 示 
”数据 接收 完毕 。status 属性 的 值 比较 多 ， 常 用 的 是 200， 表 示 请 求 成 功 。 一 般 这 两 个 属性 一 
| 起 来 判断 。 其 语法 格式 如 下 : 


xmlhttp.onreadystatechange = function(){ 
ifreadysate 一 4 and statues 一 200){ 
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} | 
(5) 当 响 应 页 处 理 结 束 后 ， 即 满足 了 “readystate-4 and status 一 200” 这 个 条 件 ， | 会 内 
就 可 以 使 用 xmlhttprequest 对 象 的 属性 获取 响应 页 的 值 了 。 常 用 的 有 responseText、 | 一 一 
responesXML、responseStream 等 ， 这 里 以 responseText 为 例 进行 介绍 。 


responesText 属性 是 将 响应 页 的 输出 信息 作为 字符 串 返 回 。 其 语法 格式 如 下 : 
str = xmlhttp.responseText; | 
(6) 最 后 使 用 send0 方 法 来 接收 回应 。send0 方 法 可 以 传递 数据 ， 但 这 取决 于 open0 | 
方法 中 的 rmethod 参数 ， 当 参数 值 为 get0 时 ， 数 据 是 附 在 URL 中 进行 传递 的 ， 当 参数 值 | 
为 post0 时 ， 数 据 只 能 使 用 send() 方 法 进行 传递 。send() 方 法 的 语法 格式 如 下 : | 
xmlhttp.send([rdate]): 


1.6.3 ”验证 码 技术 

验证 码 技术 ， 是 为 了 防止 用 户 名 和 密码 被 暴力 破解 ， 在 登录 页 面 上 生成 一 组 随机 数 ， 
每 次 刷新 页 面 时， 随机 数 都 会 改变 。 一 般 是 4 位， 也 有 更 多 位 的 。 

1， 获取 验证 码 


PHP 中 的 验证 码 可 以 通过 rand0 函 数 生成 随机 数 的 方式 得 到 。rand0 函 数 可 以 获取 指定 
范围 内 的 随机 数 ， 其 语法 格式 如 下 : | 


int rand([int min, int max]) 


如 果 和 省略 两 个 参数 ,那么 将 返回 0~RAND_MAX 之 间 的 随机 整数 ; 否则 , 返回 min~ | 
max 之 间 的 整数 。 例 如 ， 本 例 中 要 获取 4 位 十 六 进 制 的 整数 ， 代 码 如 下 : | 
01 <?php 
02 for($i=0:$i<4:$i++H){ 
03 S$num .= dechex(rand(0,15)); // 生 成 随机 数 
04 } 


06 ?> 


函数 dechex0 可 以 将 参数 转换 为 十 六 进 制 表 示 。 | 
使 用 JavaScript 也 可 以 生成 十 六 进 制 随机 数 ， 但 是 稍 有 些 复杂 。JavaScript 中 不 能 直接 | 
将 十 进 制 数 转换 为 十 六 进 制 ， 需 要 手动 进行 转换 。 首 先 使 用 Math randomO 函 数 生成 0~15 | 
之 间 的 随机 数 ， 然 后 使 用 Math ceil0 函 数 将 随机 数 取 整 ， 接 下 来 就 要 逐次 判断 该 值 ， 如 果 | 
该 值 大 于 9， 那 么 将 10~15 的 数 一 一 对 应 转换 为 a、b…， 一 直到 f。 转换 完成 后 ， 将 值 黑 | 
加 ， 最 后 传 给 valcode php 页 。 | 
使 用 JavaScript 生成 十 六 进 制 随机 数 的 完整 代码 如 下 : 
01 ”// 生 成 随机 数 


Sm 
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02 function showvalO{ 


03 num 三 ": 

04 for(i=0:i<4:i++){ /循环 输出 4 位 验证 码 

05 tmp 二 ”Math.ceil((Mathrandom() * 15)); /取得 1 位 十 六 进 制 的 整数 

06 if(ttmp > 9){ /依次 判断 随机 数 

07 switch(tmp){ 

08 case(10): // 如 果 随 机 数 等 于 10， 转 换 为 a 
09 num += "a'; 

10 break; 

11 case(11): 

12 num +='b'; /如 果 随 机 数 等 于 11， 转 换 为 b 
13 break 

14 case(12): 

15 num + 一 cs /如 果 随 机 数 等 于 12， 转 换 为 
16 break; 

17 case(13): // 如 果 随 机 数 等 于 13， 转 换 为 d 
18 num +='d'; 

19 break; 

20 case(14): /如 果 随 机 数 等 于 14， 转 换 为 e 
21 num + 一 'e': 

22 break: 

23 case(15): // 如 果 随机 数 等 于 15， 转 换 为 f 
24 num + 一 f; 

25 break; 

26 

2 jelse{f 

28 num += tmp: 

29 

30 } 

31 $('chkid").src='valcode.php?num='+num; ”// 将 生成 的 随机 数 传 给 图 像 生成 页 

32 $('chknm').value = num; // 将 随机 数 的 值 保存 到 页 面 的 隐藏 域 中 
23 


2. 显示 随机 数 图 片 


显示 随机 数 的 方式 很 多 ， 将 随机 数 写 入 一 个 图 片 中 再 显示 是 目前 比较 常用 的 一 种 。 在 
PHP 中 ， 可 以 使 用 GD 函数 库 来 实现 。 其 中 用 到 的 函数 主要 有 imagecreate() 函 数 、 
imagecolorallocate() 函数 、imagestring() 函数 、imagesetpixel() 函 数 、imagepng0) 函数 和 


imagedestroy() 函 数 。 
(1) imagecreate() 函 数 


imagecreate() 函 数 用 来 创建 一 个 基于 调 色 板 的 空白 图 像 源 ， 这 是 生成 图 片 的 第 一 步 。 


其 语法 格式 如 下 : 
resource imagecreate ( int width, int height ) 


参数 width 和 height 分 别 指定 了 图 像 的 宽 和 高 。 
(2) imagecolorallocate() 函 数 


imagecolorallocate() 函 数 可 以 为 创建 后 的 图 像 分 配 颜 色 。 其 语法 格式 如 下 : 


int imagecolorallocate ( resource image, int red, int green, int blue ) 
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参数 说 明 如 下 。 
image: 是 一 个 图 像 源 。 | 
回 red、green 和 blue: 表示 红 、 黄 、 蓝 三 元 素 的 成 分 ， 每 种 颜色 的 取 值 范围 在 1 一 | 


f 


255 之 间 。 Le 


(3) imagestring(0) 函 数 
图 像 创建 完成 后 ， 就 可 以 使 用 imagestring0 函 数 来 添加 图 像 文 字 了 。 其 语法 格式 如 下 : 


bool imagestring ( resource image, int font, int x, int y, string s, int col ) 


参数 说 明 如 下 。 

回 image: 是 一 个 图 像 源 。 

回 font: 用 于 设置 字体 ， 如 果 使 用 系统 默认 字体 ， 可 以 使 用 1 一 5 的 数字 。 | 
回 x 和 y: 分 别 表 示 文 字 相 对 于 整 幅 图 像 的 x 轴 和 y 轴 坐 标 ， 即 所 输入 的 字符 串 的 | 

> | 

回 要 显示 的 字符 串 。 

回 col: 字体 颜色 ， 也 是 使 用 imagecolorallocate() 函 数 来 分 配 。 

(4) imagesetpixel0 函 数 
使 用 imagecolorallocate0 函 数 创建 的 是 一 个 单一 背景 色 的 图 像 ， 如 果 项 望 向 图 像 中 添加 十 | 
可 以 使 用 imagesetpixel0 函 数 ， 该 函数 的 作用 是 画 一 个 像素 点 。 其 语法 格式 如 下 : 

bool imagesetpixel ( resource image, int x, int y, int color ) 

参数 说 明和 imagestring(O 函 数 相似 ， 这 里 不 再 歼 述 。 

(5) imagepng0 函 数 

该 函数 将 创建 完成 的 图 片 以 PNG 的 格式 输出 。 其 语法 格式 如 下 : 

bool imagepng ( resource image [, string filename] ) 

参数 说 明 如 下 : 
回 image: 要 保存 的 图 像 源 。 
回 filename: 要 保存 的 图 像 名 。 如 果 省 略 ， 则 直接 输出 到 浏览 

(6) imagedestroy0 函 数 

图 像 保存 完毕 后 ， 使 用 imagedestroy0 函 数 来 释放 内 存 。 其 语法 格式 如 下 : 


bool imagedestroy ( resource image ) 


验证 码 的 生成 文件 是 valcode php， 其 完整 代码 如 下 。 


01 <?php 

02 header("content-type:image/png"): // 设 置 页 面 编 码 

03 $num=S$_ GET[mum']; // 获 取 超 链接 传递 的 随机 数 

04 Simagewidth=60; // 定 义 画 布 的 宽 

05 Simageheight=18; // 定 义 画 布 的 高 

06 $numimage = imagecreate($imagewidth, $imageheight); // 创 建 画布 

07 imagecolorallocate($numimage,240,240,240); // 设 置 画 布 颜色 
08 for($i=0:$i<strlen($num):$i++){ 1/ 循环 读 取 随机 数 
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$x = mt rand(1,8)+$imagewidth*$1/4; 
$y = mt rand(1,$imageheight/4); 


My S$color=imagecolorallocate($numimage,mt rand(0,150),.mt rand(0,150),mt rand(0,150)); 
// 定 义 图 像 的 颜色 
12 imagestring($numimage,5,$x,$y,$num[$i],$color); // 将 随机 数 写 入 到 画布 中 
13 } 
14 for($i=0:$i<200:8i++){ //for 循环 语句 生成 干扰 线 
15 S$randcolor=imagecolorallocate($numimage,rand(200,255),rand(200,255), 
rand(200,255)): // 定 义 颜色 
| 16 imagesetpixel($numimage,rand09%670,rand09%620,Srandcolon; /生成 干扰 线 
T7 } 
18 imagepng($numimage): // 生 成 图 像 
19 imagedestroy($numimage); // 释 放 资 源 
20 ?> 


1.6.4 ”E-mail 激活 技术 


为 了 防止 恶意 注册 ,现在 有 很 多 网 站 采用 了 E-mail 激活 技术 。 当 用 户 注册 成 功 后 ， 并 


| 不 是 马上 就 可 以 使 用 , 而 是 需要 先 登录 邮箱 , 通过 系统 发 出 的 E-mail 中 的 超 链接 进行 激活 ， 
， 只 有 激活 后 ， 账 号 才 可 以 使 用 。 对 于 找 回 密码 的 用 户 ， 系 统 会 将 新 密码 发 送 到 邮箱 中 。 这 
| 在 一 定 程度 上 提高 了 安全 性 。 


本 模块 使 用 ZendFramework 中 的 Zend_ Mail 组 件 完成 邮件 的 发 送 操作 。 在 本 书 的 资源 


| 中 没有 提供 Zend_Mail 组 件 ， 如 果 要 运行 这 个 程序 ， 需 要 读者 自己 下 载 Zend 组 件 ， 并 将 
| 其 复制 到 本 模块 的 根 目录 下 ， 即 mr/01 文件 夹 下 。 


| 说明: 


在 执行 邮件 的 发 送 时 ， 笔 者 使 用 的 是 搜狐 的 邮箱 ， 邮 箱 名 称 是 mrsoft8888@sohu.com， 


| 密码 是 tmrsoft8888。 读 者 在 完成 用 户 注册 之 后 ， 可 以 登录 这 个 邮箱 完成 注册 用 户 的 激活 操 
| 作 ， 和 否则 注册 的 用 户 是 不 能 够 使 用 的 。 


注册 用 户 激活 的 原理 是 控制 数据 库 中 用 户 信息 表 (tb_member) 中 active 字段 的 值 。 

默认 注册 成 功 后 该 字段 的 值 是 0， 但 是 此 时 该 用 户 是 不 能 够 登录 网 站 的 ， 必 须 通过 邮箱 激 
活 , 将 该 字段 的 值 更 新 为 1 之 后 , 此 用 户 才 可 以 使 用 。 激 活 操作 的 文件 存储 于 activation.php 
文件 中 ， 其 关键 代码 如 下 : 

01 <?php 

02 session start(); 

03 header('Content-Type:text/html:charset=gb2312"): 

04 include_once("conn/conn.php"); 


05 if(lempty($ GET['name']) && lis null($ GET['name']){ /激活 注册 用 户 
06 $num=$conne->getRowsNum 

07 ("'select * from tb member where name=".$ GET['name']." and password=".$ GET[pwd].""); 
08 (Snum>0){ 

09 Supnum=$conne->uidRst 


10 ("update tb member set active = 1 where name=".$ GET[mame']." and password = 
mS_ GET[pwd'].""): 


*。30 。 
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11 if($upnum > 0){ 

12 $_SESSION['name'] = $_GET['name']: 

13 echo "<script>alert( 用 户 激活 成 功 ! );window.location href='main php';</script>"; 
14 jelse{ 

15 echo "<scripf>alert( 您 已 经 激活 ! )windowlocation href='main php';</script>"; 
16 } 

17 jelse{ 

18 echo "<script>alert( 用 户 激活 失败 ! ):windowlocation href-Yegisterphp':</scripf>" 
19 a 

20 } 

2 ?> 


1.6.5 ”应 用 键盘 响应 事件 验证 信息 是 否 合法 


发 人 员 必须 掌握 的 一 项 技术 。 下 面 就 来 简单 了 解 一 下 键盘 事件 。 


用 户 通过 onkeydown 和 onkeyup 事件 来 触发 响应 事件 。 使 用 方法 和 onclick 事件 类 似 。 
onkeydown 表示 当 键 盘 上 的 键 被 按 下 时 触发 ，onkeyup 和 它 正 好 相反 ， 表 示 当 键盘 上 的 键 | 


被 按 下 又 抬 起 时 触发 。 在 页 面 中 加 载 事 件 的 方式 有 多 种 ， 这 里 介绍 两 种 最 常用 的 方式 。 
(1) 将 事件 直接 添加 到 页 面 元 素 中 
这 种 方法 最 直接 、 简 单 。 格 式 如 下 : 
<script type="text/javascript"> 


function referO{ 


} 
</script> 
<input type="text" onkeydown="refer0" /> 
当 用 户 输入 完 信息 后 ， 按 任意 键 ，onkeydown 事件 即 被 触发 ， 并 调用 refer(0) 函 数 。 
(2) 通过 window.onload 加 载 
当 页 面 被 载 入 时 ， 事 件 被 载 入 。 代 码 如 下 : 
<script> 
window.onload = function0{ 
document.getElementById(lgname').onkeydown = functionO{ 


上 
</script> 


<input 1d="lgname" type="text" /> 
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新 用 户 注册 时 ， 随 着 信息 的 输入 ， 系 统 会 实时 显示 信息 的 正确 性 ; 用户 登 录 时 ， 无 须 
使 用 鼠标 来 选择 文本 框 ， 只 要 按 下 Enter 键 ， 焦 点 就 会 自动 下 移 。 这 些 功 能 都 是 通过 | 
JavaScript 脚本 中 的 键盘 响应 事件 来 实现 的 。 随 着 Ajax 技术 的 流行 ，JavaScript 已 经 成 为 开 | 


f 


Note 
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当 用 户 输入 信息 时 ， 每 输入 一 个 字母 ， 都 将 触发 该 事件 。 在 该 事件 调用 的 函数 中 ， 对 
| 用 户 输入 信息 进行 判断 。 例 如 ， 用 户 名 必须 大 于 等 于 2， 密码 最 短 6 位 ，E-mail 必须 合法 
| 等 。 下 面 看 一 段 验 证 用 户 名 的 代码 : 


| 01 ”// 验 证 用 户 名 
| 02 /为 记 等 于 regname 的 页 面 元 素 添加 onkeyup 事件 


‘Note 03 ”// 用 户 每 按 一 次 键 ， 都 会 调用 一 次 该 函数 
| 04 S$(‘regname').onkeyup = function |{ 
05 name = $('‘regname').value; // 获 取 输 入 内 容 
06 cname2 ="; 
| 07 if(name.match(/^[a-zA-Z ]*/) =—"){ // 判 断 输 入 字符 是 否 在 有 效 范围 之 内 
| 08 S$(‘namediv').innerHTML = '<font color=red> 必 须 以 字母 或 下 划 线 开头 </font>'; 
| 09 cnamel ="; 
| 10 }else if(name .length < 2){ // 判 断 输 入 的 字符 长 度 
| 11 SOnamediv).innerHTML = '<font color=red> 注 册 名 称 必 须 大 于 等 于 2 位 </font>'; 
| 12 cnamel ="; 
| 13 jelsef 
| 14 S$(‘namediv').innerHTML = '<font color=green> 注 册 名 称 符合 标准 </font>'; 
| 15 cnamel = 'yes'; 
| 16 } 
| 17 chkreg0); 
| Is 
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| 该 段 代码 的 运行 结果 如 图 1.12 所 示 。 


图 1.12 验证 信息 合法 性 
使 用 onkeydown 事件 还 可 以 实现 对 特定 键 的 控制 ,包括 Enter 键 、Ctrl 键 、Alt 键 等 所 


| 有 的 按键 。 这 是 通过 在 onkeydown 事件 中 使 用 keyCode 属性 来 实现 的 。keyCode 属性 能 

”知道 用 户 按 下 的 是 哪个 键 ， 例 如 Enter 键 等 于 13， 空 格 键 等 于 32 等 。 使 用 keyCode 属性 
| 的 一 般 格式 如 下 

| <script> 

| window.onload = function(){ 

| document.getElementById(lgname') = function|{ 

这 eventkeyCode 一 13){ // 判 断 用 户 是 否 按 下 了 Enter 键 


网 Ce 


第 ] 章法 册 登录 模块 SS 


在 本 模块 中 ， 实 现 了 焦点 自动 下 移 功能 。 当 用 户 按 下 Enter 键 时， 页 面 的 焦点 自动 移 


当前 光标 位 置 


回 车 后 光标 位 置 


图 1.13 焦点 自动 下 移 的 运行 效果 
1.6.6 在 PHP 中 操作 Cookie 技术 


Cookie 的 作用 是 当 用 户 第 一 次 访问 某 服务 器 时 , 服务 器 将 一 些 信息 保存 到 客户 端 计算 | 
内 。 在 以 后 的 一 段 时 间 内 ， 当 用 户 再 次 访问 这 个 服务 器 时 ， 服 务 器 通过 Cookie 信息 能 | 
够 很 快 识别 该 用 户 。 在 很 长 一 段 时 间 内 ，Cookie 都 被 当 作 一 个 不 安全 的 因素 ， 很 少 再 被 使 | 
， 转 而 使 用 Session。 但 是 ，Cookie 的 功能 确实 十 分 方便 ， 如 自动 登录 、 网 站 统计 等 ， | 
都 要 比 其 他 的 实现 方式 简单 、 快 捷 得 多 。Cookie 只 是 一 个 文本 文件 ， 不 能 访问 本 地 硬盘 ， | 
无 法 传播 病毒 、 木 马 程序 。 唯 一 要 注意 的 地 方 是 ，Cookie 只 能 识别 计算 机 ， 而 不 会 在 意 谁 
在 使 用 。 | 

在 PHP 中 操作 Cookie, 主要 用 到 setcookie0 函 数 和 $_COOKIE 预定 义 变量 。setcookie() | 
函数 的 语法 格式 如 下 : | 
bool setcookie ( string name [, string value [, int expire [, string path [, string domain [, bool secure]]]]] ) 
参数 说 明 如 下 。 
name: 设置 了 Cookie 的 名 称 。 
value: Cookie 的 值 ， 即 参数 name 的 值 。 | 
expire: 用 来 设置 Cookie 的 过 期 时 间 。 该 参数 以 时 间 戳 的 形式 存在 。 一 般 设置 | 
Cookie 过 期 时 间 的 时 候 ， 通 过 “time0+ 秒 数 ” 来 实现 。 例 如 time()+60x10， 表 示 | 
Cookie 将 在 10 分 钟 后 失效 。 如 要 设置 Cookie 马上 失效 ， 可 以 将 时 间 设 为 当前 日 | 
期 之 前 ， 如 time0-1， 那 么 Cookie 会 立即 失效 。 | 
加 ”path: 表示 Cookie 在 服务 器 端的 有 效 范围 。 如 果 path 设 为 “/”， 那 么 Cookie 在 | 
整个 服务 器 内 都 有 效 ， 如 果 Cookie 为 /05/， 那 么 Cookie 只 在 服务 器 下 的 05 目录 | 
有 效 。 | 
domain: 指定 了 Cookie 有 效 的 域名 范围 。 以 wwwmrbccd.com 为 例 , 如 果 domain | 
设 为 “.mrbced.com”， 那 么 Cookie 在 该 域 的 所 有 子 域 都 有 效 : 如 果 domain 为 | 
“www.mrbccd.com”， 那 么 Cookie 只 在 www.mrbccd.com 域内 有 效 。 | 
secure: 表明 是 否 在 https 传送 。 如 果 为 True， 表 明 Cookie 仅 在 HTTPS 连接 中 被 | 
设置 ;默认 是 False。 | 
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1.6.7 在 JavaScript 中 操作 Cookie 技术 


在 JavaScript 中 可 以 通过 document 对 象 中 的 Cookie 属性 对 Cookie 进行 创建 、 读 取 、 
删除 等 操作 。 先 来 看 一 下 创建 Cookie 的 格式 (代码 中 加 粗 的 部 分 是 需要 用 户 设置 的 )。 
documen.cookie = "Name=Value: expires=time; path=URL: domain= Domain" 
其 中 各 个 参数 的 含义 和 PHP 中 的 基本 相同 ， 这 里 就 不 再 介绍 了 。 
在 JavaScript 中 读 取 Cookie 有 一 点 复杂 。 在 JavaScript 看 来 , Cookie 就 是 一 个 字符 串 。 
例如 一 个 名 为 Count、 值 是 10 的 Cookie， 在 JavaScript 中 的 格式 如 下 。 
count=10 
如 果 有 多 个 Cookie， 那 么 其 格式 如 下 : 
count=10; name=mr; *** 


而 真正 需要 的 值 ， 其 实 只 有 10 而 已 。 所 以 ， 要 获取 Cookie 的 值 ， 就 要 使 用 split0 函 
数 将 Cookie 进行 拆 分 。 获 取 count 值 的 代码 如 下 : 


01 <script> 


02 if(document.cookie=—"){ //Cookie 等 于 空 时 

03 document.write(no cookie); 

04 }else{ 

05 cookiearr = document.cookie.split('; ); /以 分 号 为 分 隔 符 ， 对 Cookie 进行 分 割 
06 leng = cookiearr.length; 1/ 获取 数组 长 度 

07 for(i=0; i<leng: i++){ 

08 icookiearr[il.split(=)[0] == 'count){ /判断 哪 一 个 Cookie 等 于 count 
09 document.write(cookiearr[i].split(=)[1]); 。 /输出 count 的 值 

10 } 

11 } 

12 } 

13 </script> 


1.6.8 用 户 自动 登录 技术 


自动 登录 的 原理 是 : 当 用 户 打开 登录 页 面 时 ， 登 录 页 面 首先 判断 用 户 客户 端 机 器 中 的 
Cookie 值 。 如 果 该 值 不 存在 , 或 者 Cookie 已 失效 ， 将 显示 登录 表单 ; 如 果 Cookie 值 存在 ， 
则 直接 使 用 Cookie 中 所 保存 的 用 户 名 进行 登录 。 

在 本 模块 中 ， 用 户 首先 打开 的 是 index.php 页 ， 该 页 判断 Cookie 值 ， 如 果 没 有 ， 则 跳 
转 到 login.php 进行 登录 ; 如 果 Cookie 有 值 ， 则 将 Cookie 中 的 用 户 名 保存 到 Session， 直 
接 进 入 主页 。 实 现代 码 如 下 : 


01 <?php 


02 session_ start(); // 开 启 Session 支持 
03 header('Content-Type:text/html:charset=gb2312"); // 设 置 页 面 编 码 
04 if(!empty($_COOKIE['name"]) and !is_null($ COOKIE['name"]){ /判断 Cookie 是 否 


为 空 
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05 $_ SESSION[mname'] = $ COOKIE[mame']: /将 Cookie 保存 到 Session 中 | 

06 header('location:http://localhost/model/05/01/main.php");// 跳 转 到 main.php 页 | 

07 jelse{ //Cookie 为 室 ， 说 明 没有 登录 | 

08 header('location:http://localhost/model/05/01/login.php");// 跳 转 到 login.php 页 | 

09 } | 

- 4 
这 里 $_COOKIE[mame"] 的 值 是 在 登录 成 功 时 设置 的 ， 相 关 代码 如 下 。 Note 

01 <?php | 

02 soe /如 果 登 录 成 功 | 

03 setcookie('name', $name,time()+60*10); // 保 存 $name 的 同时 设置 时 间 

04 Es 


05 ?> | 
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分 页 显示 模块 
( 铝 。 自学 视频 、 源 程序 ， 配套 资源 mA2\) 


在 网 站 中 ， 当 面 对 数 据 库 中 的 大 量 数据 要 输出 时 ， 在 头脑 中 闪 过 的 第 一 个 
想法 就 是 应 用 什么 分 页 技术 输出 数据 库 中 的 数据 才 更 加 合理 、 简 单 、 快 捷 。 分 
页 功能 是 每 个 网 站 都 必 不 可 少 的 。 在 本 模块 中 ， 将 介绍 几 种 分 页 的 实现 方法 ， 
希望 对 读者 以 后 程序 的 开发 有 所 帮助 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 
内 容 。 


DI 自 定义 函数 严 换 超 长 文本 中 的 将 殊 字 符 
PI 自 定义 函数 控制 超 长 文本 的 输出 

由 通过 PDO 操作 My5QL 数据 库 

MH PHP 超 长 文本 分 页 

扣 Ajax 无 刷新 分 页 

WI 分 页 类 分 页 

MH PHP 上 下 分 页 

MH PHP 跳 转 分 页 


第 和 章 分页 显示 模块 


2.1 分 页 显示 概述 


2.1.1 模块 概述 四 会 4 


分 页 显示 模块 是 网 站 开发 中 最 常用 的 一 个 典型 模块 ， 其 最 大 用 途 就 是 将 海量 数据 或 者 直 | JNote 
长 文本 数据 分 页 输出 , 这 样 一 来 不 仅 使 页 面 的 整体 效果 更 加 紧凑 ,而 且 能 给 用 户 带 来 最 好 的 | | 
体验 。 | 
2.1.2 ”功能 结构 

在 本 模块 中 ， 整 体 上 分 为 两 部 分 内 容 ， 一 部 分 是 面向 对 象 的 分 页 ， 另 一 部 分 是 面向 过 
程 的 分 页 。 在 面向 对 象 分 页 中 ， 通 过 Smarty 模板 完成 网 页 的 动静 分 离 ， 通 过 PDO 操作 | 
MySQL 数据 库 ， 并且 将 Smarty 模板 的 配置 、 PDO 连接 操作 数据 库 以 及 分 页 方法 都 封装 到 | 
类 中 , 通过 类 中 方法 的 调用 完成 各 种 操作 ; 在 面向 过 程 分 页 中 , 完成 超 长 文本 的 分 页 、Ajax | 
无 刷新 分 页 、 跳 转 分 页 和 上 下 分 页 。 本 模块 的 系统 功能 结构 如 图 2.1 所 示 。 | 
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图 2.1 分 页 显示 模块 功能 结构 图 
2.1.3 ”程序 预览 


在 分 页 显示 模块 中 展示 了 6 种 不 同 的 分 页 方法 : 超 长 文本 的 分 页 显示 、Ajax 无 出 新 分 | 
页 、 标 准 分 页 、 跳 转 分 页 、 上 下 分 页 和 通过 分 页 类 分 页 。 | 
超 长 文本 的 分 页 显示 : 实现 对 文本 文件 中 数据 的 分 页 显示 , 主要 应 用 在 注册 须知 、| 
公司 简介 、 版 权 说 明 等 文本 类 的 文件 输出 中 ， 如 图 2.2 所 示 。 | 
回 Ajax 无 刷新 分 页 : 实现 数据 库 中 数据 的 无 刷新 显示 。 这 种 分 页 方法 主要 应 用 在 一 | 

些 支 持 视 频 播放 的 网 站 中 ， 通 过 Ajax 无 刷新 进行 分 页 ， 不 会 影响 到 视频 文件 的 | 
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播放 ， 如 图 2.3 所 示 。 
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图 2.2 超 长 文本 的 分 页 显示 
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2.3 Ajax 无 刷新 分 页 的 应 用 


| 回 上 下 分 页 : 针对 当前 的 页 面 ， 实 现 上 一 页 和 下 一 页 的 跳 转 ， 也 属于 一 种 常用 的 分 
| 页 方法 ， 如 图 2.4 所 示 。 
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图 2.4 上 下 分 页 
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第 了 章 分 页 里 示 模 块 i 


通过 分 页 类 分 页 : 将 一 种 定义 好 的 分 页 方法 封装 到 一 个 类 中 ， 当 需要 进行 分 页 操 
作 时 ， 可 以 直接 调用 这 个 类 文件 ， 并 设置 相应 的 参数 ， 即 可 实现 分 页 的 操作 ， 无 | 
须 再 重新 编写 分 页 方法 ， 如 图 2.5 所 示 。 这 是 一 种 比较 实用 的 分 页 方法 。 | 
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图 2.5 通过 分 页 类 分 页 


2.2 面向 对 象 分 页 


通过 分 页 类 实现 分 页 功能 是 一 种 非常 好 的 方法 。 一 个 分 页 类 创建 成 功 后 ， 就 可 以 直接 | 
调用 类 文件 实现 分 页 功能 ， 而 无 须 再 重新 编译 。 在 分 页 类 模块 中 ， 采 用 Smarty 模板 完成 | 
网 页 的 动静 分 离 ， 通 过 PDO 完成 对 数据 库 的 连接 和 操作 。 通 过 分 页 类 实现 分 页 的 效果 如 
图 2.6 所 示 。 
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图 2.6 通过 分 页 类 分 页 的 运行 效果 
2.2.1 Smarty 模板 的 安装 和 配置 


首先 ， 将 Smarty 模板 的 压缩 包 解 压 ， 复 制 libs 文件 夹 到 服务 器 mr\2\01\system 目录 下 。 | 
接 下 来 ， 配置 Smarty 模板 。 在 确定 Smarty 模板 文件 的 位 置 之 后 ， 在 smarty 目录 下 | 
新 建 4 个 目录 一 一 templates、templates c、configs 和 cache， 用 于 存储 不 同 的 文件 。 | 
最 后 ， 封 装 Smarty 模板 的 配置 类 ， 完 成 Smarty 模板 的 配置 操作 。 该 类 存储 于 | 
2\01\system\system.smarty.inc.php 文件 中 ， 其 代码 如 下 : | 


01 <?php 


T 


Ee L_ 


VAIO 


02 require("libs/Smarty.class.php"); // 包 含 模板 文件 

03 class SmartyProject extends Smarty{ // 定 义 类 ， 继 承 模板 类 
04 function SmartyProjectO{ // 定 义 方法 

05 Sthis->template_ dir ="./"; 1/ 指定 模板 文件 存储 位 置 
06 S$this->compile_dir = "./system/templates_c/"; // 指 定编 译文 件 存储 位 置 
07 Sthis->config dir ="./system/configs/"; /指定 配置 文件 存储 位 置 
08 $this->cache dir = "./system/cache/"; /指定 缓存 文件 存储 位 置 
09 } 

100 

NS 


2.2.2 ”加 载 PDO 数据 库 抽象 层 


在 本 模块 中 ,通过 PDO 数据 库 抽象 层 连接 和 操作 MySQL 数据 库 。PDO 是 与 PHP 5.1 
一 起 发 行 的 ， 默认 包含 在 PHP 5.1 中 。 由 于 PDO 需要 PHP 5 核心 面向 对 象 特性 的 支持 ， 
此 其 无 法 在 PHP 5.0 之 前 的 版 本 中 使 用 。 

默认 情况 下 ，PDO 在 PHP 5.2 中 为 开启 状态 ， 但 是 要 启用 对 某 个 数据 库 驱 动 程序 的 支 
持 ， 仍 需要 进行 相应 的 配置 操作 。 

在 Linux 环境 下 ， 要 使 用 MySQL 数据 库 ， 可 以 在 configure 命令 中 添加 如 下 选项 


--with-pdo-mysql=/path/to/mysql/installation 
在 Windows 环境 下 ，PDO D0 在 php.ini 文件 中 进行 本 年 ， 如 如 图 2.7 所 示 。 


川 文件 口 “ 编 弓 (6” 格 zx(O) 坦 看 W 帮助 
extension=php_ 富 目 dl1 
extension=php_pdo_mysql. dll 
;extension=php_pdo_firebird. dll 
extension=php_pdo_mssql. dll 


extension=php_pdo_oci. dll 
extension=php_pdo_oci8. dll 
extension=php_pdo_odbc. dll 
;extension=php_pdo_pgsql. dll 
extension=php_pdo_sqlite. dll 


图 2.7 在 Windows 环境 下 配置 PDO 


要 启用 PDO， 首 先 必须 加 载 “extension=php_pdo.dll”。 要 想 让 其 支持 某 个 具体 的 数据 
库 ， 那 么 还 要 加 载 对 应 的 数据 库 选 项 。 例 如 ， 要 支持 MySQL 数据 库 ， 则 需要 加 载 
“extension=php _pdo mysqldll” 选 项 。 


注意 : 

在 完成 数据 库 的 加 载 后 ， 要 保存 php.ini 文件， 并 且 重 新 启动 Apache 服务 器 ， 修 改 才 
能 够 生效 。 有 关 PDO 连接 和 操作 MySQL 数据 库 的 方法 已 经 在 关键 技术 中 作 了 详细 讲解 ， 
这 里 不 再 次 述 。 


2.2.3 ”分 页 类 模块 的 页 面 设计 
由 于 本 模块 的 开发 使 用 的 是 Smarty 模板 技术 ， 因 此 模块 的 页 面 设计 和 程序 的 开发 是 
。40 。 


| 


第 2 章 分 页 显示 模块 一 KS) 


完全 分 离 的 , 模块 的 页 面 设计 是 在 index.tpl 文件 中 完成 的 , 而 程序 的 开发 则 是 在 index.php 
文件 中 完成 。 | 
分 页 类 模块 的 页 面 设计 主要 就 是 将 程序 开发 中 模板 变量 中 的 数据 进行 输出 ， 包 括 图 | 


书 名称 、 图 书 介绍 、 出 版 时 间 和 作者 ， 以 及 分 页 功能 的 实现 。 关 键 代 码 如 下 : | 全 放 
01 {ifSisbbs—"T"} <!-- 首 先 通过 模板 变量 $isbbs 判断 是 否 有 数据 --> 
02 ”<!-- 通 过 section 内 建 函数 ， 循 环 输出 数组 模板 变量 $arraybbs 中 的 图 书 数据 --> Note 


03 {php} $1; {/php} | 
04 {section name=bbsid loop=$arraybbs} 


05 <tr> | 

06 <td height="22" class="STYLE4" {php} if($i%2==0){ echo "bgcolor=#F5F5F5";} | 
{/php}>&nbsp; | 

07 <a href="index.php?id={$arraybbs[bbsid].tb_book id}" target="_blank"> | 

08 {unhtml content=$arraybbs[bbsid]. tb_book name}</a> | 

09 </td> | 

10 <td class="STYLE4" {php} if($1%2==0){ echo "bgcolor=#F5F5F5";} {/php}> 

11 <div align="center">{$arraybbs [bbsid].tb_book synopsis}</div></td> 

12 <td height="22" class="STYLE4" {php} if($i962—0){ echo"bgcolor=#F5F5F5":} {/php}> 

3 <div align="center"> {$arraybbs[bbsid].tb_print_date}</div></td> 

14 <td height="22" class="STYLE4" {php} if($1902—0){ echo "bgcolor=#F5F5F5":} {/php}> 

15 <div align="center"> {$arraybbs[bbsid].tb_book author}</div></td> 

16 </t> 


17 {php} $it+; {/php} 
18 {/section} 
19 <table width="100%" height="22" border="0" align="center" cellpadding="0" cellspacing="0"> 


20 <tr><!-- 输 出 分 页 功能 的 信息 --> 

21 <td class="STYLE4">&nbsp: {$showpage}</td> 
22 </t> 

23 </table> 

24 


Wi 
25 <!-- 判 断 当 没有 数据 时 ， 则 输出 如 下 内 容 --> 
26 {if $isbbstell=="F" && $isbbs=="F"} 
27 <table width="96%" height="30" border="0" align="center" cellpadding="0" cellspacing="1" 
bgcolor="#58BAE9"> 


28 <tr> 

29 <td bgcolor="#FFFFFF"><div align="center"> 

30 ”<font color="# 丁 F0000">{ldelim} 该 分 类 暂 无 论坛 公告 或 帖子 {rdelim}</font></div></td> | 
31 </t> | 
32 </table> | 
33  { 僻 | 


2.2.4 ”分 页 类 模块 的 程序 开发 


分 页 类 模块 的 程序 开发 是 在 index.php 文件 中 进行 的 ， 是 一 个 纯粹 的 PHP 代码 页 。 在 | 
该 文件 中 调用 类 文件 ， 实 现 与 数据 库 的 连接 ， 读 取 数 据 库 中 的 数据 :调用 分 页 类 ， 实 现 数 | 
据 的 分 页 显示 ， 并 且 定 义 模板 变量 ， 用 于 在 模板 页 中 输出 数据 。 关 键 代码 如 下 : | 


01 <?php 
02 header("Content-Type:text/html:charset=gb2312"); 
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| 09 
数据 库 中 数据 


03 
04 
05 
06 
07 
08 
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require_once("system/system.inc.php"); // 调 用 指定 的 文件 
Sarraybbstell=$admindb->ExecSQL("select * from tb_mr_book",$conn); /执行 select 查询 语句 
if(!$arraybbstelD) { 

$smarty->assign("isbbstell","F");// 判 断 如 果 执 行 失败 则 输出 模板 变量 isbbstell 的 值 为 F 
}else{ 

$smarty->assign("isbbstell","T");// 判 断 如 果 执行 成 功 ， 则 输出 模板 变量 isbbstell 的 值 为 了 

S$smarty->assign("arraybbstell", $arraybbstell); // 定 义 模板 变量 arraybbstell， 输 出 


| 10 
| 11 iflisset($_GET["page"]){ 
| 12 S$page=$ GET["page"]; 
| 13 J}else{ 
| 14 $page=1; 
| 157 
| 16 S$arraybbs=$seppage->ShowData("select * from tb_mr book",$conn,1,$page); /调用 分 页 
| 类 ， 实 现 分 页 功能 
| 17 ifl(!$arraybbs){ 
| 18 S$smarty->assign("isbbs","F"); 
| 19 Jyelse{ 
| 20 S$smarty->assign("isbbs","T"); 
| 2 $smarty->assign("showpage",$seppage->ShowPage(",","",","")); // 定 义 输出 分 页 数据 的 模 
， 板 变量 showpage 
| 22 $smarty->assign("arraybbs",$arraybbs): 
23000 
24 $smarty->display("index.tpl"); // 指 定 连 接 的 模板 页 
25 ?> 


至 此 ， 分 页 类 模块 的 讲解 完毕 ， 程 序 的 完整 代码 参见 配套 资源 中 的 相应 内 容 。 


2.3 超 长 文本 分 页 


,2.3.1 超 长 文本 分 页 功能 概述 


利用 PHP 的 超 长 文本 分 页 显示 功能 , 可 将 指定 文本 文件 中 的 数据 进行 分 页 输出 , 效果 


图 2.8 所 示 。 
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2.8 PHP 超 长 文本 分 页 显示 的 运行 效果 
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2.3.2 超 长 文本 分 页 功能 实现 过 程 


PHP 超 长 文本 的 分 页 显示 主要 应 用 的 是 自 定义 函数 unhtml0 和 msubstr0， 有 关 这 两 个 | 
自 定义 函数 的 创建 参见 2.8.1、2.8.2 节 ， 这 里 不 再 更 述 。 会 
PHP 超 长 文本 分 页 显示 的 操作 是 在 02umr_copyrightphp 文件 中 完成 的 。 首先 ,判断 变 | 大]~ 
量 的 值 是 否 为 空 ， 并 调用 指定 的 包含 文件 ， 获 取 自 定义 函数 。 其 代码 如 下 : Note 
01 ifisset($_ POST[page])){ | 
02 Spage=$ POST['page']; | 
03 }else ifisset($_GET[page])){ | 
04 Spage=$_GET['page']; | 
05 }else{ | 
06 $page=1; | 
0o70 | 
08 include("function.php"); // 包 含 自 定义 函数 


然后 ， 通 过 file_get_contents0 函 数 获取 fle txt 文本 文件 中 的 数据 ， 返 回 到 一 个 字符 串 | 
中 ， 通 过 unhtml0 函 数 普 换 字符 串 中 的 特殊 字符 ， 通 过 strlen0 函 数 获取 字符 串 的 长 度 ， 计 | 
算 将 文本 文件 中 的 数据 分 成 几 页 进行 输出 ;通过 msubstr0 函 数控 制 数据 的 输出 ;调用 substr0 | 
函数 获取 当前 页 输出 的 内 容 。 代 码 如 下 : | 


01 <?php 

02 if($page){ | 
03 Scounter=file_get_contents("file/file.txt");// 读 取 文本 文件 中 的 数据 ， 返 回 一 个 字符 串 | 
04 S$length=strlen(unhtml($counter)); /计算 字符 串 的 长 度 ， 蔡 换 特 殊 字 符 | 
05 Spage_count=ceil($length/1050); // 对 字符 串 进行 分 页 处 理 

06 $c=msubstr($counter,0,($page-1)*1050); // 通 过 自 定义 函数 获取 上 一 页 中 的 数据 

07 $cl=msubstr($counter,0,$page*1050); // 通 过 自 定义 函数 获取 当前 页 中 的 数据 | 
08 echo substr($cl,strlen($c),strlen($c1)-strlen($c)); /获取 到 当前 页 中 要 输出 的 数据 | 
09 } | 
10 Ts 


最 后 ， 创 建 分 页 超 链 接 ， 实 现 首页 、 上 一 页 、 下 一 页 和 尾 页 之 间 的 跳 转 。 代 码 如 下 : 


01 <td width="42%" align="center" valign="middle" bgcolor="#FFFFFF"> | 
02 <span class="STYLE1">&nbsp:&nbsp: 页 次 : <?php echo Spage:?> / <?php echo | 
$page_count:?> 页 </span></td> 


03 <td width="58%" height="28" align="left" valign="middle" bgcolor="#FFFFFF"> | 
04 ”<span class="STYLE1"> &nbsp: 分 页 : | 
05 <?php | 
06 if(Spage!=1){ | 
07 echo "<a href-index.php?lmbs= 明 日 版 权 声明 &page=1> 首 页 </a>&nbsp:"; | 
08 echo "<a href-index.php?lmbs= 明 日 版 权 声明 &page=".(Spage-1)."> 上 一 页 | 
</a>&nbsp:": | 
09 } | 
10 这 $page<$page_counb{f | 
11 echo "<a href=index.php?lmbs= 明 日 版 权 声 明 &page=".($paget+1)."> 下 一 页 | 
</a>&nbsp;"; 
12 echo "<a href-index.php?lmbs= 明 日 版 权 声明 &page=".$page_count."> 尾 页 
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</a>"; 
13 } 
14 Ee 
15 </span> </td> 


| 至 此 ，PHP 超 长 文本 的 分 页 显示 介绍 完毕 。 
2.4 Ajax 无 刷新 分 页 


2.4.1 Ajax 无 刷新 分 页 功能 概述 


Ajax 无 刷新 分 页 的 强大 功能 体现 最 为 明显 的 地 方 就 是 在 播客 中 。 在 播客 中 , 在 播放 视 
| 频 文 件 的 同时 ， 如 果 使 用 的 不 是 无 刷新 分 页 ， 那 么 当 执行 翻 页 的 操作 后 ， 视 频 文件 将 被 重 
| 新 打开 ， 如 果 使 用 的 是 无 刷新 分 页 ， 就 不 会 出 现 这 种 情况 。 
为 了 更 好 地 理解 Ajax 无 刷新 分 页 的 强大 功能 ， 这 里 创建 了 一 个 视频 文件 ， 并 且 应 用 
| Ajax 无 刷新 分 页 输出 文本 文件 中 的 内 容 , 验证 无 刷新 分 页 的 操作 是 否 对 视频 文件 的 播放 有 
| 影响 。 运 行 效果 如 图 2.9 所 示 。 


er AQGE | 国 
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图 2.9 ”Ajax 无 刷新 分 页 的 应 用 
2 4.2 Ajax 无 刷新 分 页 功能 实现 过 程 


| Ajax 无 刷新 分 页 由 mr_synopsis.php、mr_synopsis_ok.php 和 discuss jsjs 3 个 文件 组 成 。 
| 在 mr_synopsis php 文件 中 , 首先 创建 一 个 div 标签 ,设置 其 id="synopsis"; 然后 在 div 
标签 中 编写 超 长 文本 分 页 输出 的 代码 ， 并 且 在 超 长 文本 分 页 的 超 链接 中 应 用 onclick 事件 ， 
| 调 上 自 定义 函数 artpagination0) ， 传 递 的 参数 为 另外 一 个 实现 分 页 功能 的 文件 

mr_synopsis_ok.php。 其 关键 代码 如 下 : 
| 01 <div id="synopsis"> <!-- 创 建 div 标签 ， 用 于 获取 JS 文件 中 返回 的 分 页 结果 -> 

02 <table width="425" border="0" cellspacing="0" cellpadding="0"> 

03 <tr> 

04 <td height="20">&nbsp:</td> 
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05 </tr> 

06 <tr> 

07 <td> 

08 <?php 

09 “// 读 取 超 长 文本 中 的 数据 ， 实 现 超 长 文本 中 数据 的 分 页 显示 
10 if($page){ ee 
11 $counter=file_get_contents("file/mr synopsis.txt"); 

12 Slength=strlen($counter); Note 
13 S$page_count=ceil($length/950); | 

14 $c=msubstr($counter,0,($page-1)*950); | 

15 $cl=msubstr($counter,0,$page*950); | 

16 echo substr($cl,strlen($c),strlen($cl)-strlen($c)); | 

I } | 

18 ?> </d> | 

19 </tr> | 

20 <tr> | 

21 <td> 


2 <table border="0" align="center" cellpadding="0" cellspacing="0"> 

23 <tr><!-- 设 置 超 长 文本 分 页 显示 的 超 链接 --> 

24 <td width="42%" align="center" valign="middle" bgcolor="#FFFFFF"> 

25 <span class="STYLE1">&nbsp:&nbsp: 页 次 : <?php echo $page;?> / <?php echo 
$page_count;?> 页 </span></td> 


26 <td width="58%" height="28" align="left" valign="middle" bgcolor="#FFFFFF"> 
27 <span class="STYLE1"> &nbsp: 分 页 : 

28 <?php 

29 if($page!=1){ 

30 ?> 


31 ”<!-- 调 用 artpagination 函数 ， 实 现 无 刷新 的 分 页 输出 --> 

32 <a href="#" onclick='return artpagination("mr_synopsis_ok.php?page=1")> 首 页 </a>&nbsp; 

33 <a href="#" onclick='return artpagination("mr synopsis_ok.php?&page=<?php echo $page-1:?>")> 
上 一 页 </a>&nbsp; 


34 <?php } 
35 if($page<$page_count){ 
36 ?> 


3 <a href="#" onclick='retum artpagination("mr_synopsis_ok.php?page=<?php echo $paget+1;?>")> 
下 一 页 </a>&nbsp: 

38 <a href="#" onclick='retum artpagination("mr synopsis ok.php?page=<?php echo 
S$page_count;?>") 人 > 尾 页 </a> 


39 <?php 

40 } 

41 ?> 
42 </span> </td> 

43 </tr> 

44 </table></td> 

45 </tr> 

46 </table> 

47 </div> 


最 后 , 通过 js 文件 中 的 自 定义 函数 artpagination() 将 mr_synopsis_ok.php 文件 中 的 分 页 
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0 
和 
结果 和 返回 到 div 标签 synopsis 中 ， 实 现 无 刷新 分 页 的 功能 。js 文件 中 的 关键 代码 如 下 : 
| 01 ”// 使 用 XMLHttpRequest 对 象 创建 异步 HTTP 请 求 


| 02 function artpagination(urD{ // 创 建 自 定义 函数 ， 获 取 传递 的 参数 
次 | 03 xmlHttp.open('get',url,true); // 通 过 get 方 法 , 执行 另外 一 个 实现 分 页 功能 的 文件 
食 秘 | 04 xmlHttp.onreadystatechange = function0O{ 
| 05 if(xmlHttp.readystate 一 4 && xmlHttp.status == 200){ /将 结果 返回 到 div 标签 
Note synopsis 中 
| 06 document.getElementById("synopsis").innerHTML 三 
| xmlHttp.responseText; 
| 07 } 
| 08 } 
| 09 xmlHttp.send(nul]): 
| 10 } 
| 说明: 
| 在 mr_synopsis_ok.php 文件 中 完成 的 分 页 功能 与 mr_synopsis.php 文件 中 是 相同 的 , 代 
， 码 不 再 重复 。 
2.5 标准 分 页 


| 利用 PHP 的 标准 分 页 功能 ， 可 在 当前 页 面 中 显示 指定 的 记录 ; 也 可 以 10 页 为 一 个 单 
| 位 ， 实 现 上 10 页 和 下 10 页 的 跳 转 ;此 外 ， 还 可 实现 当前 页 中 显示 的 10 个 超 链接 的 任意 跳 
转 。 其 运行 效果 如 图 2.10 所 示 。 


fT 明日 利于 出 这 为 页 位 加 入 收 浊 去 


2.51 PHP 标准 分 页 功能 概述 
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2.5.2 “PHP 标准 分 页 功能 实现 过 程 


在 PHP 标准 分 页 中 ,每 页 显示 5 条 数据 。 分 页 跳 转 可 以 分 为 3 个 部 分 : (1) 根据 查询 
”结果 ， 通 过 while 语句 在 当前 页 面 中 显示 5 条 记录 信息 ; (2) 以 10 页 为 一 个 单位 ， 实 现 上 
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pea (3) 在 当前 页 中 ， 在 显示 的 10 个 超 链接 中 任意 跳 转 。 PHP 标准 分 | 
是 在 mr_book.php 文件 中 完成 的 ， 其 具体 操作 步骤 如 下 。 | 
(1) 连接 数据 库 ， 定 义 分 页 中 要 使 用 的 变量 ， 包 括 数据 库 的 连接 文件 connphp、 当 | 


前 页 值 的 变量 $page 和 当前 10 页 值 的 变量 $link_type。 关 键 代码 如 下 : | 食 内 
01 <?php 
02 include_once("conn/conn.php"); Note 
03 ifisset($_POST[page])){ | 
04 Spage=$ POST[page']; | 
05 yelseif(isset($ GET['page']){ | 
06 Spage=$ GET['page']; | 
07 jelse{ | 
08 $page=1; | 
09 } | 
10 if(isset($_ GET["link type])) { 

11 Slink type=$ GET["link type]: 
12 }else{ 
13 Slink type=0: 
Ta 
15 if(isset($_GET[vv])) { 
16 S$vv=$_GET[vv]: 
17 }else{ 
18 $vv=0; 
19 } 
20 include("function.php"); 
21 ?> 

说 明 : 


因为 本 模块 应 用 switch 语 和 完成 网 页 的 框架 操作 ， 所 有 功能 部 包含 在 index.php 文件 | 
中 ， 所 以 将 上 述 的 这 段 代码 存 储 在 index.php 文件 中 ， 作 为 公共 内 容 。 | 


(2) 在 mr book.php 文件 中 ， 从 数据 库 中 读 取 数据 ， 统 计 出 总 的 记录 数 ， 设 置 每 页 
示 的 记录 数 ， 计 算出 共有 几 页 ， 并 通过 while 语句 循环 输出 数据 库 中 的 数据 。 代码 如 下 


01 <?php | 
02 if($page){ // 判 断 当 前 页 变量 的 值 是 否 存在 | 
03 Spage_size=5; /定义 每 页 输出 的 记录 数 | 
04 $query="select * from tb_forum send ": // 编 写 查 询 语句 , 应 用 count 统计 总 的 记录 数 | 
05 $result=$conn->query($query); // 执 行 查询 语句 | 
06 S$res=$result->fetchAll(PDO::FETCH_ASSOC): | 
07 $message_count=count($res); | 
08 S$page_count=ceil($message_count/$page_size);// 计 算出 总 共有 几 页 | 
09 Soffset=($page-1)*$page_size: // 输 出 上 一 页 结束 的 记录 数 | 
10 $query_2=$conn->query("select * from tb_forum send limit $offset, $page_size"); | 
11 foreach($query 2 as $value){ | 
全 | 
13 <tr> | 
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14 <td width="35%" align="center" bgcolor="#FFFFFF"> 
| 15 <a href="index_ok.php?send id=<?php echo S$value['tb_send 1d'];?>" 
| target="_blank"> 
| 16 <?php echo $value['tb_send_subject];?></a> 
| 17 </td> 
| 18 <td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value['tb_send date']:;?></td> 
| 19 <td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value['tb_send user];?></td> 
vote | 20 <td width="10%" align="center" bgcolor="#FFFFFF"> 
| 21 <?php 
| 22 $query_s=$conn->query("select * from tb _forum restore "); 
| 3 echo count($query_s); 
| 24 ?> 
| 25 </td> 
| 26 </t> 
| 27 <?php 
| 28 
| 29 } 
30 ?> 


(3) 创建 分 页 超 链 接 ， 实 现 分 页 中 以 10 页 为 单位 的 跳 转 ， 以 及 当前 10 页 中 不 同 页 
面 之 间 的 跳 转 。 关 键 代 码 如 下 : 


| 01 <table width="800" height="30" border="1" cellpadding="]l" cellspacing="1" 
| bordercolor="#FFFFFF" bgcolor="#D0DODO"> 


| 02 <tr> 

| 03 <td height="22" align="center" bgcolor="#FFFFFF"> 

| 04 ”<span class="STYLE2"> 帖 子 统计 : <?php echo $message_count;?> 条 &nbsp; </span></td> 
| 05 <td align="center" bgcolor="#FFFFFF"> 每 页 显示 : <?php echo $page_size; ?> 条 </td> 
| 06 <td align="center" bgcolor="#FFFFF"> 共 有 : <?php echo $page_count;?> 页 </td> 

| 07 <td align="center" bgcolor="#FFFFFF"><span class="STYLE2"> 分 页 : 

| 08 -<?php 


09 ”Snext=$link type*10; /以 10 页 为 一 个 单位 

| 10 $n=$link type-1; /上 一 个 10 页 的 值 

| 11 ”Sm=S$link type+l; ”// 下 一 个 10 页 的 值 

| 12 ”Sprev_page=$page-10;// 当 前 页 的 上 10 页 的 值 


| 13 Smm="; 

| 14 ”if($link type=-0){ ”// 判 断 变 量 的 值 是 否 等 于 0， 如 果 等 于 0 则 输出 首页 图 标 , 无 超 链接 
| 15 echo "<img sre=\"images/02.jpg\" width=\"8\" height=\"9\" border=\"O\" tile=\" 首 页 >" 

| 16 J}else{ // 否 则 ， 为 首页 图 标 设置 超 链 接 ， 连 接 到 首页 ， 并 为 上 10 页 的 图 标 设置 超 链接 
| 17 echo "<a href='index.php?lmbs= 明 日 论坛 &vv=0&link_type=0&page=1'> 


| 18 <img src=\"images/02.jpg\" width=\"8\" border=\"0\"” height=\"9\"” border=\"0\" title=\" 首 页 
| ></a>&nbsp;"; 


| 19 $ccc=$vv-10: /定义 变量 ， 变 量 $vv 是 当前 10 页 的 起 始 位 置 ，$ccc 是 上 一 个 
10 页 的 起 始 位 置 

| 20 echo "<a hre 人 >"'index.php?lmbs= 明 日 论坛 &vv=S$ccc&link type=$n&page=$prev_page'> 

| 21 <img src=\"images/01.jpe\" width=\"8\" height=\"9\" border=\"0\" title=\" 上 十 页 ></a>"; 

2 } 

| 2300 > 

| 24 <?php 


25 。“// 通 过 for 循环 语句 ， 输 出 当前 页 中 的 10 个 超 链接 
二 。48 。 


2 > 模块 | 
第 之 章 分 页 显示 模块 一 $9 | 


26 for(9j=1:9j<=$page count:Sj++) { // 根 据 统 计 的 总 的 记录 数 循 环 输出 | 
27 Spnext=Snext+9j; // 定 义 当前 页 中 输出 的 页 码 值 | 
28 if(S$mm=—=10){ // 判 断 当 $mm 的 值 等 于 10 时 跳出 循环 | 
29 break: | 
3 让 4 
31 if($mm>$page_ coun){ // 判 断 当 $mm 的 值 大 于 总 记录 时 跳出 循环 | ~ 
32 break; 
3 } | Note 
34 if($page_count-$vv<10){ | 
35 if($mm>=$page count-$vv){ 1/ 判断 当 Smm 的 值 大 于 等 于 总 记录 数 减 去 当 | 
前 10 页 的 起 始 值 时 ， 跳 出 循环 | 
36 break:; | 
37 } | 
38 } | 
39 echo "<a hreE-index php?lmbs= 明 日 论坛 &vv=S$vv&link type=$link type &page =$pnext>" | 
40 过 $page 一 S$pnext){ /为 当前 单 击 的 页 码 描 红 | 
41 echo "<span class='STYLE1'> $pnext </span>"; | 
42 jelse{ | 
43 echo " $pnext "; // 输 出 当前 10 页 超 链接 中 的 数据 | 
44 } | 
45 echo "</a>"; | 
46 Smm=$mmtl; | 
CA | 
48 > | 
49 </span><span class="STYLE2"> | 
50 ”<?php | 
51 $vv=$vvt$mm; // 定 义 变量 的 值 ， 定 义 下 10 页 的 起 始 位 置 | 
52 这 $page_count-$vv<=0){ /判断 当 总 记录 数 小 于 等 于 下 10 页 的 起 始 位 置 时 ， 输 出 尾 页 的 图 标 | 
53 echo "<img ste=\"images/03 jpe\" width=\"8\" height=\"9\" border=\"O\" tide=\" 尾 页 W>"; | 
54 J}else{f // 否 则 输出 下 10 页 的 超 链 接 | 
55 echo "<a href="index.php?lmbs= 明 日 论坛 &vv=$vv&link type=$m&page= $pnext> | 
56 <img sre=\"images/03.jpg\" width=\"8\" height=\"9\" border=\"O\" title=\" 下 十 页 ></a>"; | 
S70 | 
58 ”if($message_count==0){ /判断 当 总 记录 数 等 于 0 时 ， 输 出 没有 记录 | 
59 echo "没有 记录 !"; | 
60 1} | 
61 > | 
62 </span>&nbsp;&nbsp;</td> | 
63 </tr> | 
64 </table> ! 


至 此 ， 有 关 PHP 标准 分 页 的 功能 介绍 完毕 ， 详 细 代 码 参见 配套 资源 中 的 相应 内 容 。 
2.6 跳 转 分 页 | 


2.6.1 PHP 跳 转 分 页 功能 概述 


PHP 跳 转 分 页 的 特点 是 能 够 实现 从 当前 页 面 到 其 他 指定 页 面 的 跳 转 ， 运 行 效果 如 
。49 。 


EC pp 型 护 决 靖 角 
图 2.11 所 示 。 多 数 情况 下 它 与 标准 分 页 功能 结合 使 用 ， 以 增加 分 页 操作 的 灵活 性 。 
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图 2.11 PHP 跳 转 分 页 的 运行 效果 


2.6.2 PHP 跳 转 分 页 功能 的 实现 过 程 


在 PHP 跳 转 分 页 中 ， 每 页 显示 5 条 数据 。 分 页 跳 转 可 以 分 为 3 个 部 分 : (1) 通过 文本 


框 实现 指定 页 面 的 跳 转 功能 ;2) 以 10 页 为 一 个 单位 ,实现 上 10 页 和 下 10 页 的 跳 转 ;(3) 
| 在 当前 页 中 ， 在 显示 的 10 个 超 链接 中 任意 跳 转 。PHP 跳 转 分 页 是 在 mr_forum.php 文件 中 
| 完成 的 ， 其 具体 操作 步骤 如 下 。 


(1) 连接 数据 库 ， 定 义 分 页 中 要 使 用 的 变量 ， 包 括 数据 库 的 连接 文件 conn.php、 当 


”前 页 值 的 变量 Spage 和 当前 10 页 值 的 变量 $link_type。 关 键 代码 如 下 : 


01 <?php 
02 include_once("conn/conn.php"); 
03 iflisset($_POST['page']){ 


04 Spage=$_POST['page']; 
05 jelse if(isset($_GET['page']){ 
06 Spage=$_ GET['page']; 
07 jelse{ 

08 S$page=1; 

09 } 

10 if(isset($ GET['link type'])) { 
11 Slink type=$_GET['link type']: 
12 jelse{ 

1 S$link type=0: 

1200: 

15 if(isset($_GET[vv'])) { 

16 $vv=$_GET[vv']: 

17 jelse{ 

18 $vv=0; 

T9003 

20 include("function.php"); 

| ?> 
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第 了 章 分 页 里 示 模 块 


说 明 : 
因为 本 模块 应 用 switch 语句 完成 网 页 的 框架 操作 ， 所 有 功能 都 包含 在 index.php 文件 
中 ， 所 以 将 上 述 的 这 段 代码 存储 在 index.php 文件 中 ， 作 为 公共 内 容 。 


(2) 在 mr_forumphp 文件 中 ， 从 数据 库 中 读 取 数据 ， 统 计 出 总 的 记录 数 ， 设 置 每 页 
显示 的 记录 数 ， 计 算出 共有 几 页 ， 并 通过 while 语句 循环 输出 数据 库 中 的 数据 。 代 码 如 下 : 


01 <?php | 
02 if(Spage){ // 判 断 当前 页 变量 的 值 是 否 存在 | 
03 Spage_size=5; /定义 每 页 输出 的 记录 数 | 
04 $query="select * from tb_forum send "; /编写 查询 语句 ， 应 用 count 统计 总 的 记录 数 | 
05 Sresult=$conn->query($query); // 执 行 查询 语句 | 
06 Sres=Sresult->fetchAll(PDO::FETCH_ASSOC): | 
07 $message_count=count($res): | 
08 $page_count=ceil($message_count/$page_size); 。 // 计 算出 总 共有 几 页 | 
09 Soffset=($page-1)*$page_size; // 输 出 上 一 页 结束 的 记录 数 | 
10 $query_2=$conn->query("select * from tb_forum send limit $offset, $page_size"); | 
11 foreach($query_ 2 as $value){ | 
12 ?> | 
13 <tr> | 
14 <td width="35%" align="center" bgcolor="#FFFFFF"> | 
15 <a href="index_ok.php?send id=<?php echo $value['tb_send id"]:?>" | 
target="_blank"> | 
16 <?php echo $value['tb_send subject"];?></a> | 
Ti </td> | 
18 <td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value['tb send datel:?></td> | 
19 <td width="25%" align="center" bgcolor="#FFFFFF"><?php echo $value['tb_send_user];?></td> | 
20 <td width="10%" align="center" bgcolor="#FFFFFF"> | 
21 <2php | 
22 $query_s=$conn->query("select * from tb_forum restore "); | 
23 echo count($query_s); | 
24 ?> | 
25 </td> | 
26 </tr> | 
27 <?php | 
28 } 
2 
30 ?> 


(3) 创建 一 个 表单 ， 用 于 提交 跳 转 的 页 码 值 ， 并 且 通 过 JavaScript 脚本 判断 提交 的 值 
是 否 有 效 。 代 码 如 下 : 


01 <form name="forml" method="post" action="index.php?lmbs= 明 日 论坛 " onSubmit="return 


chkO"> 
02 <input type="submit" name="Submit" value=" 跳 转 ">&nbsp; 
03 <input name="page" type="text" size="3"> 
04 <input type="hidden" name="pages" value="<?php echo $page_count:;?>"> 
05 </form> 
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判断 提交 的 页 码 值 是 否 有 效 是 通过 自 定义 函数 chk0 完 成 的 ， 关 键 代码 如 下 。 


01 
02 


<script language="javascript"> 
function chkO{ 
if(form].page.value<=0 || parseInt(form] .page.value)>parseInt(form!.pages.value){ 

alert(" 您 输入 的 页 码 无 效 !1"); 
form!1 .page.focus(); 
return(false); 

} 

return(true); 


</script> 


| (4) 编写 PHP 语句 ， 实 现 分 页 中 以 10 页 为 单位 的 跳 转 ， 以 及 当前 10 页 中 不 同 页 面 
| 之 间 的 跳 转 。 关 键 代码 如 下 : 


12 


29 


的 起 始 值 时 ， 跳 出 循环 


30 
31 


<td align="center" bgcolor="#FFFFFF"><span class="STYLE2"> 分 页 : 


<?php 

S$next=$link type*10: // 以 10 页 为 一 个 单位 
$n=S$link type-1; /上 一 个 10 页 的 值 
Sm=$link type+1; /下 一 个 10 页 的 值 
Sprev_page=$page-10; // 当 前 页 的 上 10 页 的 值 
$mm=""; 


if($link_type 一 0){ /判断 变量 的 值 是 否 等 于 0， 如 果 等 于 0 则 输出 首页 图 标 , 无 超 链接 
echo "<img sre=\"images/02.jpg\" width=\"8\" height=\"9\" border=\"0\" title=\" 首 页 >"; 
jelsef /否则 ， 为 首页 图 标 设 置 超 链接 ， 连 接 到 首页 ， 并 为 上 10 页 的 图 标 


echo "<a href='index.php?lmbs= 明 日 论坛 &vv=0&link_type=0&page=1'> 
<img src=\"images/02.jpg\" width=\"8\ border=\"0\” height=\"9\"”border=\"0\W title=\" 首 页 


S$ccc=$vv-10; ”// 定 义 变量 , 变量 $vv 是 当前 10 页 的 起 始 位 置 ，$cce 是 上 一 个 10 页 


echo "<a hre 人 ="index.php?lmbs= 明 日 论坛 &vv=S$ccc&link type=$n&page=$prev_page'> 
<img sre=\"images/01.jpg\" width=\"8\" height=\"9\" border=\"O\" tile=\" 上 十 页 ></a>"; 
<?php 
// 通 过 for 循环 语句 ， 输 出 当前 页 中 的 10 个 超 链接 
for($j=1;9j<=$page_count:Sj++){ /根据 统 计 的 总 的 记录 数 循环 输出 


S$pnext=$next+9j; /定义 当前 页 中 输出 的 页 码 值 

if($Smm—10){ // 判 断 当 $mm 的 值 等 于 10 时 跳出 循环 
break: 

} 

if($mm>$page_count){ // 判 断 当 $mm 的 值 大 于 总 记录 数 时 跳出 循环 
break: 

} 


if($page_count-$vv<10){ 
if($mm>=$page_count-$vv){ // 判 断 当 $mm 的 值 大 于 等 于 总 记录 数 减 去 当前 10 页 


break; 
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32 } | 

33 echo "<a hreF-index php?lmbs= 明 日 论坛 &vv=$vv&link type=Slink type&page =$pnext>"; | 

34 这 Spage 一 Spnexb{ /为 当前 单 击 的 页 码 描 红 | 

35 echo "<span class='STYLE1'> $pnext </span>"; | 

36 Jelse{ | # 
37 eo et /| 输出 当前 10 页 超 链接 中 的 数据 | 贸 - 
38 | 

39 echo "</a>"; Note 
40 Smm=$mmt+1; | 

41 } | 

42 ?> | 

43 </span><span class="STYLE2"> | 

44 <?php | 

45 Svw=$vvtSmm; // 定 义 变量 的 值 ， 定 义 下 10 页 的 起 始 位 置 。 

46 这 Spage countSvv<=0){ // 判 断 当 总 记录 数 小 于 等 于 下 10 页 的 起 始 位 置 时 ， 输 出 尾 页 的 图 标 | 

47 echo "<img sre=\"images/03.jpg\" width=\"8\" height=\"9\" border=\"O\" tile=\" 尾 页 >"; | 

48 Jelse{ // 否 则 输出 下 10 页 的 超 链接 | 

49 echo "<a href-index php?lmbs= 明 日 论坛 &vv=$vv&link type=Sm&page=Spnext> 

50 <img sre=\"images/03.jpg\" width=\"8\" height=\"9\" border=\"O\" title=\" 下 十 页 ></a>"; | 
5 | 

52 if($message count==0){ // 判 断 当 总 记录 数 等 于 0 时 ， 输 出 没有 记录 | 

53 echo "没有 记录 !"; 

S40 

55 ?> 


56 </span>&nbsp;&nbsp:</td> 
至 此 ， 有 关 PHP 跳 转 分 页 的 功能 介绍 完毕 ， 详 细 代码 参见 配套 资源 中 的 相应 内 容 。 


2.7 上 下 分 页 | 


2.7.1 PHP 上 下 分 页 功能 概述 


PHP 上 下 分 页 功能 以 当前 页 码 为 基础 ， 实 现 上 一 页 和 下 一 页 的 跳 转 ， 并 且 输 出 上 一 页 
中 最 后 一 条 数据 和 下 一 页 中 第 一 条 数据 的 内 容 。 运 行 效果 如 图 2.12 所 示 。 
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2.12 PHP 上 下 分 页 的 运行 效果 


。 53 。 
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2.7.2 PHP 上 下 分 页 功能 的 实现 过 程 


| 在 PHP 上 下 分 页 中 ， 每 页 显示 2 条 数据 。 分 页 跳 转 可 以 分 成 两 个 部 分 : (1) 上 下 翻 页 的 跳 
| 转 ，(2) 以 10 页 为 一 个 单位 ， 实 现 上 10 页 和 下 10 页 的 跳 转 ， 以 及 在 当前 页 显示 的 10 个 超 
人 一 ”链接 中 的 任意 跳 转 。 有 关 第 二 部 分 的 内 容 已 经 在 2.6.2 节 中 作 了 详细 介绍 ， 这 里 不 再 袭 述 。 


下 面 主要 介绍 上 下 翻 页 的 跳 转 ， 以 及 输出 上 一 页 的 最 后 一 条 数据 和 下 一 页 的 第 一 条 数 
| 据 的 方法 。 此 外 ， 还 将 讲解 分 栏 功能 的 实现 方法 。 其 操作 是 在 mr_dictionaryphp 文件 中 完 


在 输出 数据 库 中 的 数据 时 ， 应 用 的 是 分 栏 方法 ,创建 一 个 1 行 2 列 的 表格 来 输出 数据 
， 库 中 的 数据 。 实 现 的 原理 如 下 : 
| (1) 在 while 语句 之 外 ， 定 义 一 个 变量 $number=0。 
| (2) 在 while 语句 中 ， 通 过 站 语句 判断 取 模 ($number % 2) 的 值 是 否 等 于 0， 如 果 
| 等 于 0， 则 是 行 的 开始 (执行 <tr>); 如 果 不 等 于 0， 则 是 行 的 结束 (执行 <\tv>)。 
| (3) 在 while 语句 之 外 ， 执 行 $number++。 


其 关键 代码 如 下 : 
01 <?php 
02 $number=0; // 定 义 变量 
03 /省 略 了 部 分 PHP 代码 
04 $query_2=mysql query("select * from tb_mr bccd limit $offset, $page_size"); 
05 while($myrow_2=mysql fetch_ array($query_2)){ /循环 输出 
06 让 (Snumber%2) 一 0){ // 判 断 取 模 的 值 是 否 等 于 0 
07 ?> 
08 <tr> <!-- 在 取 模 的 值 等 于 0 时 执行 此 操作 --> 
09 <?php 1 2 
10 <td colspan="2" align="center"> 
<!-- 输 出 数据 库 中 数据 --> 
</td> 
11 <?php 
12 if( ($number %2) (= 0){ // 判 断 当 取 模 的 值 不 等 于 0 时 
13 ?> 
14 </tr> <!-- 当 取 模 的 值 不 等 于 0 时 执行 此 操作 --> 
15 ”<?php} // 结 束 while 循环 
16 $numbertt+:; // 变 量 值 增加 
17 > 


| 上 一 页 、 下 一 页 跳 转 功 能 的 实现 ， 关 键 就 是 对 当前 页 值 page 的 判断 。 设 置 上 一 页 的 
| 超 链 接 和 获取 上 一 页 中 的 最 后 一 条 数据 ， 关 键 代码 如 下 : 
| 0 < widb"205" align"center" bgcolor "HFFFFFF"S 上 部: <a href-"index php?imbs= 明 卓 


编程 词典 &page=<?php 

| 02 if($page=—"" or $page<=1){ 
03 echo 1; 
04 }else{ 
05 echo ceil(($page*2-2)/2): 
06 } 


第 了 章 分 页 里 示 模 块 § 


07 es 

08 <?php 
09 Spage_ 3=($page-2)*2+1; | 
10 if($page 3<=0){ | 


11 Spage_ 3=0; | 7 

i200 | 全 
13 $query_3="select * from tb_mr bccd limit $page 3,1"; I 

14 Sresult3=$conn->prepare($query_3); // 准 备查 询 语句 JVote 
15 $result3->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 | 

16 Sres3=$result3->fetch(PDO::FETCH ASSOO); ! 
17 echo S$res3['tb_bccd name']; 
18 ?7></a></td> 


获取 下 一 页 超 链 接 中 page 的 值 和 下 一 页 中 的 第 一 条 数据 ， 关 键 代码 如 下 : 


01 <td width="238" align="center" bgcolor="#FFFFFF"> 下 一 部 : <a href-"index.php?lmbs= 明 日 
编程 词典 &page=<?php 
02 if($page=="" or $page<=1){ 


03 echo 2; 

04 jelse if($page>=$page_count){ 
05 echo $page_count; 

06 }else{ 

07 echo ceil(($page*2+1)/2); 
08 }?>> 

09 <?php 


10 $page_4=($page)#2; 
11 if($page_4<=0){ 


12 S$page_4=2; 

13 

14 if($page_4=—$page_count*2){ 

15 $page_4=$page_count*2-1: 

1 

17 $query_4="select* from tb_mr bccd limit $page 4,1"; 

18 $result4=$conn->prepare($query_4); /准备 查询 语句 

19 Sresult4->execute(); // 执 行 查 询 语句 ， 并 返回 结果 集 


20 S$res4=$result4->fetch(PDO::FETCH_ASSOC); | 
21 echo S$res4['tb_bccd_name']; 
2 ?></a></td> 


PHP 上 下 分 页 中 的 其 他 内 容 已 经 在 2.6.2 节 中 介绍 过 了 ， 这 里 不 再 獒 述 ， 完 整 程序 代 | 
码 参 见 配套 资源 中 的 相应 内 容 。 


2.8 技术 提炼 | 


2.8.1” 自 定义 函数 替换 超 长 文本 中 特殊 字符 技术 


在 输出 超 长 文本 中 的 数据 时 ， 应 用 自 定义 函数 unhtml0 实 现 对 超 长 文本 中 的 特殊 字符 | 
进行 蔡 换 。unhtml0 函 数 的 语法 格式 如 下 : 
。55 。 


AI 


| TD 
入 
| So 
| function unhtml($content){ // 定 义 自 定义 函数 的 名 称 
| Scontent=htmlspecialchars($content); // 转 换文 本 中 的 特殊 字符 
| $content=str_replace(chr(13),"<br>",$content); // 蔡 换文 本 中 的 换行 符 
| $content=str_replace(chr(32),"&nbsp:",$content); // 蔡 换文 本 中 的 &nbsp: 
傅 扩 | $content=str_replace("[ [","<",$content]: // 蔡 换文 本 中 的 大 于 号 
| Scontent=str_replace(") )",">",$content); // 蔡 换文 本 中 的 小 于 号 
Note Scontent=str replace("| |"," ",$content); / 蔡 换文 本 中 的 空格 


return trim($content); /删除 文本 中 首尾 的 空格 


首先 ， 通 过 file_get_contents() 函 数 读 取 超 长 文本 中 的 数据 ， 返 回 一 个 字符 串 变 量 
| $counter。unhtml() 函 数 就 是 对 这 个 字符 串 进行 操作 ， 蔡 换 字符 串 中 的 特殊 字符 ， 然 后 再 执 
| 行 其 他 操作 。 其 关键 代码 如 下 : 


01 <?php 

02 include("function.php"); /调用 指定 的 文件 
03 if($page){ 

04 $counter=file_ get contents("file/file.txt"); // 读 取 指 定 文件 的 内 容 
05 Slength=strlen(unhtml($counter)); /1/ 获 取 字 符 串 长 度 
06 S$page_count=ceil($length/1050); // 统 计 每 页 的 字符 数 
07 $c=msubstr($counter,0,($page-1)*1050); /计算 上 页 

08 $cl=msubstr($counter,0.$page*1050); /计算 当前 页 

09 echo substr($cl,strlen($c),strlen($cl1)-strlen($c)); // 输 出 当前 页 的 内 容 
10 } 

11 ?> 


file_get_contents() 函 数 主要 是 用 于 将 指定 文件 中 的 内 容 读 入 到 一 个 字符 串 。 如 果 操作 系 
统 支 持 内 存 映 射 技术 ， 还 可 应 用 内 存 映 射 技 术 来 增强 其 读 入 文件 的 性 能 。 其 语法 格式 如 下 : 
string file_get_contents(string filename [, int use_include_path [, resource context]]) 
其 中 ， 参 数 说 明 如 表 2.1 所 示 。 
表 2.1 file_get_contents() 函 数 的 参数 说 明 


参 。 数 说 有明 
filename | ”必要 参数 。 用 于 指定 某 文件 的 路 径 及 名 称 

use include path | ” 可 选 参数 。 如 果 将 该 参数 设置 为 True，PHP 会 尝试 按照 include_pa 也 标准 
i 可 选 参 数 。 设 置 提高 文件 流 性 能 的 一 些 选项 


2.8.2” 自 定义 函数 控制 超 长 文本 的 输出 技术 


在 PHP 超 长 文本 分 页 显示 中 ， 控 制 超 长 文本 中 数据 输出 应 用 的 是 自 定义 函数 
msubstr()。 该 函数 的 语法 格式 如 下 : 
01 functionmsubstr($str,$start.Slen){ //$str 指 的 是 字符 串 ，S$start 指 的 是 字符 串 的 起 始 位 
置 ，$len 指 的 是 长 度 


02 Stmpstr—"; 
| 03 S$strlen=$start+ $len: /1/ 用 $strlen 存储 字符 串 的 总 长 度 〈( 从 字符 串 的 起 始 
”位 置 到 字符 串 末 尾 的 总 长 度 ) 
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04 for($i=0:$i<$strlen: $i++){ // 通 过 for 循环 语句 ， 循 环 读 取 字 符 串 | 
05 if(ord(substr($str,$i.1))>0xa0){ ”// 如 果 字 符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 
0xa0， 则 表示 为 汉字 | 


06 Stmpstr=substr(SstrSi2); /每 次 取出 两 位 字符 赋 给 变量 Stmpstr， 即 等 于 | 

一 个 汉字 | 贪 内 
07 Sitt; // 变 量 自 加 1 | 天 一 
08 jelse{ // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr NM 
09 Stmpstr =substr(Sstr, $1,1):} | ote 
10 } | 
11 return $tmpstr; // 输 出 字符 串 | 
12} | 


首先 ， 应 用 include 调用 指定 的 包含 文件 function.php; 然后 ， 应 用 file_get_contents() | 
函数 读 取 指 定 文本 文件 中 的 数据 ， 并 应 用 自 定义 函数 unhtml0 葵 换 字 符 串 中 的 特殊 字符 ， | 
最 后 ， 应 用 msubstr() 函 数控 制 数 据 的 输出 。 其 关键 代码 如 下 : | 
01 <?php 
02 include("function.php"); 
03 if($page){ 


04 $counter=file_get_contents("file/file.txt");// 读 取 文 本 文件 中 的 数据 ， 返 回 一 个 字符 串 
05 Slength=strlen(unhtml($counter)); 1/ 计算 字符 串 的 长 度 ， 蔡 换 特殊 字符 

06 Spage_count=ceil($length/1050):; // 对 字符 串 进 行 分 页 处 理 

07 $c=msubstr($counter,0,($page-1)*1050); // 通 过 自 定义 函数 获取 上 一 页 中 的 数据 
08 $cl=msubstr($counter,0,$page*1050); /1/ 通 过 自 定义 函数 获取 当前 页 中 的 数据 
09 echo substr($cl,strlen($c),strlen($c1)-strlen($c); /获取 到 当前 页 中 要 输出 的 数据 
10 } 

11 ?> 


2.8.3 连接 数据 库 类 的 创建 


要 实现 数据 库 中 数据 的 分 页 显示 ， 首 先 必 须 与 数据 库 建 立 连 接 。 这 里 通过 一 个 自 
定义 的 类 来 实现 数据 库 的 连接 ， 其 中 操作 数据 库 使 用 PDO 技术 。 该 类 存储 于 
02\01\system\system.class.inc.php 文件 中 ， 其 代码 如 下 : 


01 class ConnDB{ /数据 库 连 接 类 

02 var $dbtype: 

03 Var $host; | 
04 Var $user; | 
05 var $pwd: | 
06 var $dbname; | 
07 function ConnDB($dbtype,S$hostSuserSpwd.Sdbname){ // 构 造 方法 | 
08 S$this->dbtype=$dbtype; | 
09 S$this->host=$host: | 
10 S$this->user=$user; | 
11 Sthis->pwd=$pwd: | 
12 Sthis->dbname=$dbname; | 
13 } | 
14 function GetConnIdO{ /实现 数据 库 的 连接 并 返回 连接 对 象 ”| 
15 这 $this->dbtype 一 "mysql" || $this->dbtype—"mssql"){ /判断 数据 库 类 型 | 
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| 16 Sdsn—"$this->dbtype:host=$this->host;dbname=S$this->dbname";// 定 义 连接 语句 
| 1 }else{ 
| 18 $dsn="$this->dbtype:dbname=$this->dbname"; 
| 19 } 
| 20 try{ 
| 21 Sconn = new PDO(Sdsn, Sthis->user $this->pwd); /初始 化 一 个 PDO 对 象 ， 创 建 数据 
| 库 连 接 对 象 Spdo 
22 S$conn->query("set names utf8"); /设置 编码 格式 
23 Teturn $conn; /返回 连接 对 象 
24 } catch (PDOException $e) { 
25 die ("Error!: " . $e->getMessage() . "<br/>"); 
26 } 
| 2 } 
| 28  】} 


”2.8.4 操作 数据 库 类 的 创建 


| 数据 库 连接 成 功 后 ， 接 下 来 进行 数据 库 的 操作 。 这 里 同样 对 PDO 操作 数据 库 的 方法 
| 进行 了 封装 。 在 AdminDB 类 中 ， 可 以 执行 数据 的 添加 、 更 新 、 删 除 和 查询 操作 。 类 文件 
| 存储 于 2\01\system\system.class.inc.php 中 ， 其 代码 如 下 : 


01 classAdminDB{ /数据 库 管理 类 

02 function ExecSQL($sqlstr$conn){ 

03 $sqltype=strtolower(substr(trim($sqlstr),0,6)); 

04 Srs=$conn->prepare($sqlstr); // 准 备查 询 语句 

05 Srs->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 
06 if($sqltype=—"select"){ 

07 $array=$rs->fetchAll(PDO::FETCH_ASSOC);// 获 取 结 果 集 中 的 所 有 数据 
08 if(count($array)==0 || $rs 一 false) 

09 return false:; 

10 else 

11 Teturn $array; 

12 }elseif ($sqltype 一 "update" || $sqltype=="insert" || $sqltype=="delete"){ 

13 if($rs) 

14 Teturn true; 

15 else 

16 Teturn false: 

17 } 

18 } 

19 


2.8.5 ”分 页 类 的 创建 


为 了 让 分 页 技术 的 应 用 更 加 简单 ， 可 以 将 分 页 的 方法 写 入 一 个 类 中 ， 这 样 就 可 以 在 需 
| 要 使 用 分 页 的 地 方 直接 调用 这 个 类 实现 分 页 的 功能 。 分 页 类 同样 存储 于 2\01\system\ 
| system.class.inc.php 文件 中 ， 其 代码 如 下 : 

01 class SepPage{ 
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02 Var $rs; 
03 Var $pagesize; 
04 Var $nowpage; 
05 Var $array; 
06 Var $conn; 
07 Var $sqlstr; 
08 function ShowData($sqlstr, $conn, $pagesize, $nowpage){ // 定 义 方法 
09 if(!lisset($nowpage) || Snowpage—"") /判断 变量 值 是 否 为 空 有 
10 S$this->nowpage=1; // 定 义 起 始 页 | 
11 else | 
12 Sthis->nowpage=$nowpage:; | 
13 S$this->pagesize=$pagesize; // 定 义 每 页 输出 的 记录 数 | 
14 S$this->conn=$conn; // 连 接 数 据 库 返 回 的 标识 | 
15 Sthis->sqlstr=$sqlstr; // 执 行 的 查询 语句 | 
16 Soffset=($this->nowpage-1)*$this->pagesize; | 
17 $sql=$this->sqlstr." limit $offset, $this->pagesize"; | 
18 Sresult=$this->conn->prepare($sq]l); // 准 备查 询 语句 | 
19 Sresult->execute():; // 执 行 查询 语句 ， 并 返回 结果 集 | 
20 Sthis->array=S$result->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 | 
21 这 count($this->array) 一 0 || $this->array 一 false) | 
22 return false; | 
23 else | 
24 Teturn $this->array; | 
25 站 | 
26 function ShowPage($contentname,$utits,$anothersearchstr, $anothersearchstrs,$class){ | 
2 $str=""; | 
28 Sres=$this->conn->prepare($this->sqlstr); /准备 查询 语句 | 
29 Sres->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 | 
30 S$this->array=S$res->fetchAll(PDO::FETCH_ASSOC);// 获 取 结 果 集中 的 所 有 数据 | 
31 Srecord=count($this->array): // 统 计 记录 总 数 | 
32 S$pagecount=ceil($record/$this->pagesize); /计算 共有 几 页 | 
33 $str.=$contentname."&nbsp:".$record."&nbsp;".$utits."&nbsp: 每 页 &nbsp; | 


34 ".$this->pagesize."&nbsp:".$utits."&nbsp; 第 &nbsp:".$this->nowpage."&nbsp; 页 / 共 
&nbsp;".$pagecount."&nbsp; 页 "; 


35 S$str.="&nbsp;&nbsp:;&nbsp:&nbsp:"; 
36 if($this->nowpage!=1) 
37 $str.="<a hre=".$_SERVER['PHP_SELF']." 


38 ?page=1&page_type=".$anothersearchstr"&parameter2=".$anothersearchstrs." 
class=".$class."> 首 页 </a>"; 


9 else 

40 $str.="<font color=#555555> 首 页 </font>"; 
41 $str="&nbsp:": 

42 if($this->nowpage!=1) 

43 S$str.="<a href=".$_SERVER['PHP_SELF']."? 


44 page=".($this->nowpage-1)." 
45 &page type=".$anothersearchstr."é:parameter2=".$anothersearchstrs." class=".$class."> 上 一 页 </a>"; 


46 else 
47 $str.="<font color=#555555> 上 一 页 </font>"; 
48 Sstr.="&nbsp;"; 
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49 if($this->nowpage!=$pagecount) 
50 S$str="<a href=".$_SERVER['PHP_SELF']."?page=".($this->nowpaget+1)." 
51 &page type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 下 一 页 </a>"; 
52 else 
53 $str.="<font color=#555555> 下 一 页 </font>"; 
54 $str="&nbsp:": 
4 if($this->nowpage!=$pagecount) 
56 $str.="<a href=".$ SERVER['PHP SELF']."?page=".$pagecount." 
57  &page type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 尾 页 </a>"; 
58 else 
59 $str.="<font color=#555555 心 尾 页 </font>"; 
60 这 count($this->array) 一 0 || $this->array= 一 false) 
61 return "无 数据 !"; 
62 else 
63 return $str; 
64 1 
65 } 


在 这 个 分 页 类 中 ， 包 含 两 个 方法 ， 即 ShowData($sqlstr,$conn,$pagesize,$nowpage) 和 


| ShowPage($contentname,$utits, $anothersearchstr,$anothersearchstrs,$class)。 其 中 , ShowData() 


| 方法 根据 参数 传递 的 SQL 语句 ， 完 成 数据 的 分 页 查询 操作 ， 并 且 返 回 查 询 结 果 ; 在 
| ShowPage() 方 法 中 ， 根 据 参数 传递 的 值 定义 分 页 超 链接 的 输出 内 容 。 


|2.8.6 ”类 的 实例 化 


类 创建 成 功 之 后 ， 要 想 在 程序 中 调用 类 中 的 方法 完成 指定 的 操作 ， 必 须 先 完成 类 的 实 


例 化 操作 ， 通 过 类 实例 化 返回 的 对 象 才 可 以 调用 指定 的 方法 。 有 关 类 的 实例 化 操作 在 本 模 
| 块 中 都 统一 存储 在 2\01\system\system.inc.php 文件 中 ， 其 代码 如 下 : 


01 <?php 

02 require("system.smarty.inc.php"); // 包 含 Smarty 配置 类 

03 require("system.class.inc.php"); // 包 含 数据 库 连接 和 操作 类 

04 $connobj=new ConnDB("mysql","localhost","root","l11","db_pagination"); 1/ 数 据 库 连接 
类 实例 化 

05 $conn=$connobj->GetConnId(): // 执 行 连接 操作 ， 返 回 连 接 标识 

06 $admindb=new AdminDB(); /| 数据库 操作 类 实例 化 

07 S$seppage=new SepPage(); // 分 页 类 实例 化 

08 S$usefun=new UseFun(); // 使 用 常用 函数 类 实例 化 

09 $smarty=new SmartyProjectO: // 调 用 smarty 模板 

10 function unhtml($params){ 

11 extract($params); 

| Stext=$content; 

13 global $usefun; 

14 Teturn $usefun->UnHtml($text):; 

1S 

16 $smarty->register_function("unhtml","unhtml");// 注 册 模 板 函 数 

> 
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上 传 与 下 载 模块 
( 名 自学 视频 、 源 程序 ， 配 套 资源 mi\3\) 


文件 的 上 传 与 下 载 是 一 个 完整 的 网 站 不 可 或 缺 的 功能 之 一 ， 如 果 不 能 提供 
这 样 的 功能 ,浏览 者 便 会 觉得 这 个 网 站 的 开发 不 是 很 完美 。 本 章 将 全 面 介 绍 PHP 
的 上 传 、 下 载 技术 ， 包 括 从 php.ini 文件 的 配置 ， 到 上 传 函 数 的 使 用 ， 到 最 后 
的 动手 实践 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


PI 文件 目录 及 文件 的 相关 操作 
WI 文件 的 上 传 

WI 文件 的 下 载 

了 文件 重 名 自动 检测 

PH 利用 下 载 码 实现 文件 搜索 

”如何 限制 用 户 的 空间 使 用 大 小 
了 文件 上 传 的 相关 配置 


.App 内 组 搞 半 精 角 
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3.1 上传、 下 载 模块 概述 


仿生 3.1.1 模块 概述 


使 用 过 ASP 的 读者 一 定 会 对 其 上 传 功能 记忆 犹 新 。 要 在 ASP 中 实现 上 传 ， 需 要 使 用 
Active 组 件 完成 ， 或 者 编写 一 大 段 代码 实现 无 组 件 上 传 。 这 对 于 新 手 来 说 ， 难 免 令 人 望 而 
| 生 航 ， 即 便 具有 一定 的 基础 也 会 有 一 定 的 难度 。PHP 在 这 方面 要 比 ASP 方便 得 多 ， 只 要 一 
| 个 上 传 函数 就 能 够 完全 实现 。 如 果 是 针对 大 文件 上 传 或 下 载 ， 也 只 需 在 php ini 配置 文件 中 
设置 几 个 参数 值 即 可 。 在 本 模块 中 将 对 PHP 的 上 传 和 下 载 功能 的 实现 进行 系统 、 全面 的 讲解。 


”3.1.2 ”功能 结构 


为 了 便于 读者 更 好 地 学 习 ， 下 面 给 出 上 传 、 下 载 模块 的 功能 结构 流程 图 。 
| 在 上 传 、 下 载 模 块 中 ， 用 户 需 要 先 注册 为 本 模块 用 户 ， 然 后 才能 登录 模块 进行 相关 操 
| 作 。 用 户 未 登录 时 ， 身 份 为 游客 ， 可 通过 下 载 码 下 载 未 公开 文件 。 用 户 成 功 登 录 后 ， 系 统 
| 将 自动 判别 访客 的 级 别 ， 级 别 不 同 ， 分 配 的 权限 也 不 同 。 根 据 级 别 的 不 同 ， 可 以 执行 上 传 
| 新 文件 、 删 除 已 上 传 个 人 文件 等 操作 ， 还 可 以 更 改 文件 公布 状态 等 。 

上 传 、 下 载 模块 的 功能 结构 如 图 3.1 所 示 。 


上 传 下 载 模块 


[ 
普通 用 户 会 员 用 户 高 级 用 户 


二 
| 


填 冯 村 | 
诗 并 四 才 
这 准 邳 吾 
卒 冷风 洲 
塞 并 好 才 

这 闪 沸 才 型 潍 才 洲 沪 
这 冷 济 洲 


图 3.1 上 传 下 载 模 块 的 功能 结构 图 
3.1.3 ”程序 预览 
上传、 下 载 模块 由 多 个 页 面 组 成 ， 下 面 列 出 几 个 典型 的 页 面 ， 其 他 的 参见 配 书 光盘 中 


| 的 程序 ， 
> SE 
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$9 
上 传 、 下 载 模块 的 登录 页 面 如 图 3.2 所 示 。 如 果 想 下 载 自己 上 传 的 文件 ， 则 首先 执行 | 
登录 操作 ， 登 录 成 功 后 才 可 以 在 个 人 信息 显示 处 单 击 “ 下 载 文件 ” 超 链 接 进入 下 载 页 。 


戏 还 光量 : mrjj | 
上 传 文件 | 下 载 文件 全 中 
用 户 名 : mxkj 您 是 [普通 用 户 ] 
密 码 : eseeee 
验证 码 : fc | 您 剩余 空间 : 2.00NB | 
[| 看 不 清 剩余 率 : 100. 00% | 
登录 注册 找 加 密码 重新 登录 
图 3.2 用 户 登 录 页 面 | 
上 传 、 下 载 模块 的 主页 如 图 3.3 所 示 。 如 果 用 户 身份 为 游客 ， 则 可 通过 下 载 码 下 载 未 | | 
公开 文件 ， 如 果 成 功 登 录 ， 则 可 以 执行 上 传 新 文件 、 删 除 已 上 传 个 人 文件 等 操作 ， 还 可 以 | 
更 改 文件 公布 状态 等 。 
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图 3.3 ”上传 、 下 载 模块 的 主页 
上 传 文件 页 面 如 图 3.4 所 示 ， 主 要 是 用 于 上 传 新 文件 到 系统 分 配 的 空间 内 。 


号 上 传 文件 


C:\Documents and settine ll [TAI ld] 
上 传 


图 3.4 上 传 文件 页 面 


下 载 文件 页 面 如 图 3.5 所 示 。 顾 名 思 义 ， 该 页 面 主要 是 展示 已 上 传 文件 的 相关 信息 ， 
户 可 以 通过 单 击 该 页 面 上 文件 名 称 的 超 链 接 下 载 所 需 文件 。 
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GR 了 tp 六 蜡 扩 志 糙 名 


TR: EE] 2: 二 了 El 


| rc I28.0 安 装 包 rer 理 乐 ”4lbhfss2bbaczcas 。 2011-02-25 08:53:28 。 公开 
优 站 | 全 造 反选 是 附 选 择 首页 上 一 页 下 一 页 尾 页 当前 是 第 ! 页 / 共 1 页 1 条 记录 | 页 


图 3.5 下 载 文件 页 面 


当前 用 户 为 游客 身份 时 ， 也 可 以 通过 文件 的 下 载 码 下 载 对 应 的 上 传 文件 ， 如 图 3.6 所 示 。 


"Dy: [tssztbsczca BN 0#: [gl 


音乐 4lbfasZbbac2eas 。 2011-02-25 09:53:28 


图 3.6 通过 下 载 码 下 载 文件 
3.2 数据 库 设计 


| 下 面 通过 实现 一 个 上 传 、 下 载 模块 ， 学 习 上 传 、 下 载 技术 在 实际 中 的 应 用 。 首 先 来 看 
| 数据 库 的 设计 和 连接 。 


3.2.1 数据 库 分 析 


| 首先 定义 上 传 下 载 模块 ， 使 用 MySQL 数据 库 ， 名 称 是 db_netu。 其 中 包含 3 个 表 ， 分 
| 别 是 tb_member (用 户 信 息 表 )、tb_upfile (上 传 文件 表 )， 以 及 tb_uptype (文件 类 型 表 )， 
| 如 图 3.7 所 示 。 其 中 ，tb_member 存储 用 户 的 相关 信息 ，tb_upfile 是 整个 上 传 、 下 载 模块 
”的 核心 数据 表 ， 记 录 了 上 传 、 下 载 文件 的 相关 信息 ，tb_uptype 主要 记录 上 传 文件 的 种 类 。 


回 服务 器 : localhost 》 国 数据 库 : 吧 netm 
办 结构 “是 sQL 万 搜 索 ”局 查 向 ” 乱 导 出 ”本 Import 多 其 作 


表 类 型 整理 说 明 
tb-member MyISAM ”9b2312-chinese_ci 用 户 列表 
tb-upfile = MyISAM ”9b2312-chinese-ci 上 传 文件 列表 


tb_uptype ”MyISAM gb2312_chinese_ci 。 文件 类 型 列表 


图 3.7 db_netu 数据 库 


”3.2.2 创建 数据 表 


1. tb_member 表 


| tb_member 表 记 录 了 注册 用 户 ， 包 括 用 户 名 、 密 码 、 密 保 问 题 、 密 保 答案 、 用 户 等 级 
”权限 以 及 用 户 个 人 空间 使 用 情况 等 信息 。tb_member 表 的 结构 及 说 明 如 图 3.8 所 示 。 
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胃 服务 器 : localhost ， 局 数据 库 : db_netu ， 国 表 :tb_member 
国 训 览 ” 回 桂 构 ”如 SOL 万 扫 过 于 项 入 陋 导 出 靖 Import 及 六 作 


字 良 大 型 整理 H Null 2 
厂 这 int(4) 理 自动 泥 码 
厂 name varchar(50) 。 gb2312_chinese_ci 理 文件 名 称 
厂 password varchar(50) gb2312_chinese_cl 理 用 户 要 码 
厂 question varchar(50) gb2312_chinese_ci EE 窗 码 保护 
mm answer varchar(50) gb2312_chinese_ci 否 窗 保 问题 
厂 email varchar(200) gb2312_chinese_cl 香 Email 
ri nt) 否 ”用户 权限 
厂 Se bilgint(s0) 再 合用 鱼网 


图 3.8 tb_member 表 结构 及 说 明 
2. tb_upfile 表 


tb_upfile 表 记 录 了 用 户 上 传 的 文件 信息 ， 包 括 文件 名 、 文 件 类 型 、 文 件 下 载 码 等 。 | 


tb_upfile 表 的 结构 及 说 明 如 图 3.9 所 示 。 
中 服务 器 : localhost ， 辐 数据 库 : db_netu  ， 国 表 : rtb_upfile 
_ 轩 浏览。 辐 结 构 “ 驴 sQL 户 棉 索 用 入 蜡 导 出 阐 Import 区 车 


字段 类 型 整理 说 明 
i Iint(4) 自动 编号 
filename varchar(200) gb2312-chinese-cl 文件 名 称 
filepath varchar(200) gb2312-_chinese_cl 。 保存 路 径 
filetype varchar(100) gb2312_chinese_cl 。 文件 型 
upauthor varchar(200) gb2312_chinese_ci 上 传 用 户 
uptime Himestamp 上 上传 时 间 
chkdownnum varchar(20) gb2312_chinese-c| 。 下 萤 码 
ispub int(D) 是 否 公开 
Mesize bigintc20) 文件 大 小 


图 3.9 tb_upfile 表 结 构 及 说 明 
3. tb_uptype 表 


tb_uptype 表 规 定 了 上 传 文件 的 类 型 ， 用 户 可 以 将 文件 上 传 到 不 同 的 类 型 中 ， 以 方便 


日 后 查找 。tb_uptype 表 的 结构 及 说 明 如 图 3.10 所 示 。 
加 服务 器 : localhost 》 怕 数据 库 : db_netu 。 ， 图 表 : tb_uptype 
唱 浏 览 。 攻 结构 “有 RsQL 万 搜索 于 疾 入 圈 导 出 “此 Import 锋 操 作 


字段 类型 整理 额外 说 明 
El int(g) auto-increment 自动 编号 
genrename varchar(200) gb2312.chinese.ci 类 型 名 和 


图 3.10 tb_uptype 表 结 构 及 说 明 
3.2.3 ”数据 库 连接 文件 


PHP 连接 数据 库 并 非 难事 ， 但 是 在 每 次 应 用 数据 库 连 接 时 都 重复 输入 类 似 代 码 ， 无 疑 
会 浪费 大 量 的 时 间 ， 同 时 也 降低 了 代码 的 书写 效率 和 可 读 性 。 所 以 ， 本 模块 中 将 数据 库 的 | 


连接 方法 封装 到 一 个 类 中 ， 这 样 在 应 用 数据 库 连接 操作 时 ， 调 用 该 类 文件 即 可 。 


conn\conn.php 文件 的 关键 代码 如 下 : 
01 <?php 
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另外 ， 


.pp 内 齐 搞 半 精 角 


class opmysql{ 
private $host = 'localhost': /服务 器 地 址 
Private $name = 'root'; /登录 账号 
Private $pwd='111' // 登 录 密码 
private $dBase = 'db_netu'; /数据 库 名 称 
Private $conn = /数据 库 链 接 资源 
Private $result ="; /结果 集 
Private $msg = /返回 结果 
private $fields; // 返 回 字段 
private $fieldsNum = 0; // 返 回 字段 数 
Private SrowsNum = 0; // 返 回 结果 数 
Private $rowsRst ="; // 返 回 单条 记录 的 字段 数组 
Private $filesArray = array(); // 返 回 字 段 数组 
private $rowsArray = array0: // 返 回 结 果 数 组 
// 初 始 化 类 
function _construct($host=",$name=",$pwd=",$dBase="){ 
if($host !{= ") 
Sthis->host = $host:; 
if($name !=") 
S$this->name = $name; 
isSpwd (=") 
Sthis->pwd = $pwd; 
if($dBase !=") 


Sthis->dBase = $dBase; 
Sthis->init_conn(); 


} 

// 链 接 数 据 库 

function init_connO{ 
Sthis->conn=(Dmysql_connect($this->host,$this->name,$this->pwd); 
@mysql_select_db($this->dBase.$this->conn); 
mysql_query("set names gb2312"); 

} private $dBase='db_netu'; /| 数据库 名 称 

/省 略 部 分 代码 


该 自 定义 类 中 还 包含 了 一 些 常 用 的 数据 库 操作 方法 。 具 体 方法 和 相关 说 明 参 见 


” 配 书 光盘 中 的 源 程序 ， 这 里 不 再 袭 述 。 
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3.3 用 户 注 册 


1 注册 模块 概述 
本 上 传 、 下 载 模块 是 一 个 公开 面向 用 户 的 个 人 网 络 存 储 空间 程序 ， 自助 式 的 用 户 注 


其 功能 模块 之 一 。 新 用 户 可 以 通过 单 击 登录 模块 中 的 “注册 ” 超 链接 来 进入 自助 式 用 


册页 


而 ， 如 图 3.11 所 示 。 
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入 2 
电子 相册 注册 页 面 


| 
| 
| 
注册 名 称 : ezkj 苦 吉 您 ， 衣 用 户 名 可 以 使 用 | 
| 
| 
| 


注册 窑 码 : [eseeee 密码 符合 要 求 。 密码 强度 : 弱 
密码 输入 正确 


确认 密码 : 


密 保 问题 : |111 
密 保 答案 : |111 


Enail: Pingrostt@ninerisoft. con 


注册 ”详细 信息 


图 3.11 用 户 注册 页 面 
3.3.2 注册 页 面 设 计 


注册 页 面 的 设计 没有 太 多 实质 性 的 理论 可 言 ， 其 主要 是 通过 表单 来 获取 用 户 注 册 的 相 
关 信息 ， 将 数据 提交 到 register_chk.php 文件 中 ， 完 成 用 户 注 册 信 息 的 处 理 。 注 册页 面 文 件 
register.php 的 相关 代码 如 下 : 


29 <div id="morediv" style="display:non:"> 
30 ”<!-- 省 略 部 分 代码 --> 
3 </table> 


01 <?php | 
02 session start(); | 
03 header('Content-Type:text/html:charset=gb2312"); | 
04 ?> | 
05 <table id="regfm" width="500" border="0" cellspacing="0" cellpadding="0" align="center"> | 
06 <tr> | 
07 <td height="25" colspan="3" align="center" valign="middle"> 电 子 相 册 注 册页 面 </td> | 
08 </tr> | 
09 <tr> | 
10 <td width="100" height="25" align="right" valign="middle"> 注 册 名 称 : </td> | 
11 <td width="200" height="25" align="left" valign="middle">&nbsp; | 
12 <input id="regname" name="regname" type="text" class="txt" /></td> | 
13 <td height="25"><div id="namediv" class="regdiv"> 名 称 由 英文 字母 和 数字 及 下 划 线 | 
组 成 </div></td> | 
14 </tr> | 
15 <tr> | 
16 <td width="100" height="25" align="right" valign="middle"> 注 册 密 码 : </td> | 
17 <td width="200" height="25" align="left" valign="middle">&nbsp: | 
18 <input id="regpwdl" name="regpwd1" type="password" class="txt" /></td> | 
19 <td height="25"><div id="pwddiv1" class="regdiv"> 请 输入 密码 </div></td> | 
20 </tr> | 
| <tr> | 
22 <td width="100" height="25" align="right" valign="middle"> 确 认 密码 : </td> | 
23 <td width="200" height="25" align="left" valign="middle">&nbsp: | 
24 <input id="regpwd2" name="regpwd2" type="password" class="txt" /></td> | 
25 <td height="25"><div id="pwddiv2" class="regdiv"> 确 认 密 码 </div></td> | 
26 </tr> | 
27 </able> | 
28 </table> | 
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</div><center> 

<div style=" text-align:center;text-align:center; width:50%; height:25px: line-height:25px;"> 
<button id="regbtn" class="btn" disabled="disabled"> 注 册 </button>&nbsp; 

<button id="morebtn" class="btn"> 详 细 信 息 </button></div></center> 


3.3.3 ”Ajax 无 刷新 技术 验证 注册 信息 


用户 的 耐心。 


| 新 用 户 注册 账号 时 ， 如 果 在 每 次 提交 注册 信息 之 后 才 返 回 不 符合 要 求 的 错误 信息 ， 不 
| 但 会 降低 执行 效率 〈 用 户 之 前 填写 的 所 有 注册 信息 都 会 随 着 操作 失败 而 消失 )， 也 会 消磨 


为 了 让 用 户 及 时 得 到 反馈 信息 ， 注 册 模 块 采用 目前 比较 流行 的 Ajax 技术 ， 


对 注册 信息 执行 无 刷新 验证 操作 。 这 里 以 验证 注册 用 户 名 为 例 进行 介绍 。 该 过 程 是 通过 js 
| 文件 类 目录 下 的 registerjs 文件 来 实现 的 。 其 关键 代码 如 下 : 


20 


function $(id){ 
return document.getElementById(id); // 获 取 标 签 ID 
由 
S$('regname').onblur = functionO{ 
name = $('regname').value; 
if(cnamel 一 'yes'"){ 
xmlhttp.open('get','chkname.php?name="+name,true):; 
xmlhttp.onreadystatechange = fonction(){ 
if(xmlhttp.readyState 一 4){ 
这 xmlhttp.status 一 200){ 
var msg = xmlhttp.responseText; 
if(msg —'1){ // 判 断 用 户 名 是 否 被 占用 
$(namediv').innerHTMIL="<font color=green> 恭 喜 您 ， 该 用 户 名 可 以 使 用 !</font>"; 
cname2 = 'yes'; 
jelse if(msg —'2){ 
SCnamediv).innerHTML="<font color=red> 用 户 名 被 


cname2 ="; 
Jelse{ 
$Cnamediv).innerHTML="<font 


cname2 = "; 
3 
} 
} 
} 
xmlhttp.send(nul]); 
chkregO|; 


} 


| 在 registerjs 文件 中 ， 无 刷新 调用 chknamephp 文件 ， 对 注册 用 户 名 进行 验证 。 根 据 
| chkname.php 文件 的 返回 值 进行 判断 ,该 用 户 名 是 否 可 用 。chkname.php 文件 的 关键 代码 如 下 : 


01 


<?php 


“68 . 


第 3 章 上 伟 与 下 载 模块 


02 session start(); /开启 会 话 

03 include_once "conn/conn.php": /加载 数据 库 配 置 文件 | 

04 Sreback = '0'; // 定 义 验证 返回 值 | 

05 $sql = "select * from tb member where name=".$ GET[mame']."": | 

06 $num= $conne->getRowsNum($sq)); // 执 行 数据 库 查 询 操作 | 贪 内 
07 ”这 num 一 D{ // 判 断 是 否 存在 匹配 结果 | 天 一 
08 Sreback = "2'; /如 果 存 在 则 将 验证 返回 值 赋值 2 

09 jelse if($num == 0){ 和 Note 
10 Sreback = "1'; // 如 果 不 存在 则 将 验证 返回 值 赋值 】 | 

11 jelse{ | 

也 Sreback = $conne->msg_error(); // 输 出 错误 信息 | 

B00 | 

14 echo Sreback:; // 输 出 验证 返回 值 | 

15 ?> 


3.3.4 注册 信息 处 理 
创建 register_chk.php 文件 ， 获 取 注 册页 面 registerphp 中 提交 的 注册 信息 ， 最 终 将 上 


户 的 注册 信息 添加 到 指定 的 数据 表 中 ， 同 时 为 新 注册 用 户 创建 一 个 属于 自己 的 存储 文件 目 
录 。 注 册 信息 处 理 文件 register_chk.php 的 相关 代码 如 下 : 

01 <?php 

02 session start(); 

03 header('Content-Type:text/html:charset=gb2312"); 

04 include_once 'conn/conn.php'; 

05 S$reback = '0'; 

06 $sql = "insert into tb_member(name,password,question,answer,email) 


07 values(".trim($ GET['name']).",".md5(trim($_GET['pwd'])).".".$_GET['question'].", 
08 "S$_GET[answer].",".S GET['email].")"; 


09 $num = $conne->uidRst($sqD: 

10 if($num 一 1){ 

11 $_SESSION[mame] = $_GET['name']: 

过 $usermame=$_ SESSION[mame']; | 
13 $dirmmame="../upfile/$usermame"; 1/ 创建 一 个 和 用 户 名 相同 的 文件 夹 “| 
14 mkdir($dirname,0700): | 
15 Sreback ="'1'; | 
16 Sfile = 'index.txt'; | 
17 Snewfile="../upfile/$username/index.php"; | 
18 if (!copy(Sfile, $newfile)) { | 
19 echo "failed to copy $file..\n"; | 
20 } | 
| jelsef | 
22 Sreback = $conne->msg_error(); | 
23 | 
24 echo S$reback: | 
25 2 ! 
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3.4 用 户 登 录 


”3.4.1 登录 模块 概述 


上 传 、 下 载 模块 通过 用 户 登 录 实现 个 人 空间 的 划分 、 控 制 上 传 文件 的 公开 或 隐藏 ， 以 


| 及 对 个 人 上 传 文件 的 管理 。 登 录 模块 的 运行 效果 如 图 3.12 所 示 。 


i 98 拘 才 


欢迎 光临 : leonsk 


上 传 文件 | 下 载 文件 


您 是 [高 级 用 户 ] 
您 剩余 空间 : 5.00WB 
稀 余 率 :100. 00% 


图 3.12 ”登录 模块 运行 效果 


| 登录 模块 中 包含 用 户 登录 、 新 用 户 注册 、 找 回 密码 等 功能 ， 同 时 输出 了 用 户 的 权限 、 
| 剩余 空间 大 小 、 剩 余 空 间 百 分 比 以 及 “上 传 文件 ”和 “下 载 文件 ” 超 链接 。 由 于 篇 幅 所 限 ， 
| 下 面 只 对 用 户 登录 模块 中 的 一 些 特殊 功能 进行 详细 讲解 ， 其 完整 内 容 参 见 配 书 光 盘 源 代码 。 


3.4.2 用 户 权限 输出 以 及 空间 使 用 计算 


| 用 户 登录 后 ， 系 统 将 输出 用 户 的 权限 以 及 用 户 个 人 空间 的 使 用 情况 。 下 面 介绍 如 何 判 
| 断 登 录用 户 的 权限 ， 以 及 计算 空间 的 使 用 情况 。 
| (1) 在 indexphp 中 ， 根 据 SESSION 变量 传递 的 登录 用 户 名 ， 从 数据 库 中 读 取 该 用 
| 户 的 个 人 信息 ， 对 tb_member 数据 表 中 right 字段 的 值 进行 判断 ， 进 而 输出 用 户 的 权限 。 
| 其 关键 代码 如 下 : 
| 01 -php 
02 Srightsql= "select * from tb member where name=".$ SESSION[mame'].""; /查询 语句 
03 Srightarr= $conne->getRowsRst(Srightsql); // 返 回 查 询 结果 集 ， 判 断 用 户 权限 


04 Sconne->close_rst(); 


05 if(Srightar['right]==2) { // 如 果 返 回 查 询 值 为 2， 判 断 为 高 级 用 户 
06 echo "<span class=\"vip1\">[ 高 级 用 户 ]</span>"; 

07 yelse if(Srightan['right]==1) { 

08 echo "<span class=\"vip2\">[ 会 员 用 户 ]</span>"; /如 果 返 回 查 询 值 为 1， 判 断 为 会 员 用 户 
09 jelse{ 

10 echo "<span class=\"vip3\">[ 普 通用 户 ]</span>"; // 否 则 判断 为 普通 会 员 
vo 
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(2) 计算 用 户 的 空间 使 用 情况 。 首 先 ， 从 数据 库 中 读 取 权限 字段 中 的 数值 ， 判 断 当 | 
前 用 户 的 登录 权限 ， 通 过 用 户 的 权限 来 设 定 其 使 用 空间 大 小 。 然 后 ， 从 数据 库 中 读 取 该 用 | 
户 的 空间 使 用 情况 ， 将 其 分 配 空间 的 大 小 与 该 用 户 已 使 用 空间 大 小 进行 差 值 运算 。 最 后 ， | 

将 取得 的 结果 进行 格式 化 输出 ， 显 示 当 前 登录 用 户 的 剩余 空间 和 剩余 空间 的 百分比 。 相 关 | 贪 内 
代码 如 下 : Ma 


01 ”<p> 您 剩余 空间 : 


02 <?php 

03 if($rightar['right]==2) { // 如 果 权 限 为 高 级 用 户 ， 则 为 其 分 配 空间 5MB 

04 这 5000000-$rightarr['content]<=1000) { // 判 断 用 户 空间 是 否 不 足 

05 echo "已 满 ! "; 

06 }else { 

07 Sformatted=(5000000-$rightarr['content])/1000000; /对 剩余 空间 大 小 进行 换算 

08 Sforstr=sprintf("%6-01.2f",$Sformatted); /对 计算 的 空间 大 小 数值 进行 格式 化 

09 echo $forstr."MB"; 

10 echo "<p> 剩 余 率 :".sprintft"%6-01.2f",((1-Srightarr[content]/5000000)*100))."%</p>"; 

11 } /对 用 户 的 剩余 空间 以 百分比 的 格式 输出 

12 jelse if($rightarr['right']==1) { 

19 这 3000000-$rightarr['content]<=1000) { 

14 echo "已 满 ! "; 

1 }else{ 

16 Sformatted=(3000000-Srightarr['content"])/1000000; 

17 Sforstr=sprintf("%-01.2f",$formatted); 

18 echo $forstr."MB"; 

19 echo "<p> 剩 余 率 :".sprintft"%6-01.2f",((1-Srightarf'content]3000000)*100))."96</p>"; 

20 3 

2l }else { 

22 if(2000000-Srightarr['content]<=1000) { 

23 echo "已 满 !"; 

24 jelse{ 

25 S$formatted=(2000000-$rightarr['content"])/1000000; 

26 Sforstr=sprintf("%-01.2f", $formatted): 

2 echo $forstr."MB"; ! 

28 echo "<p> 剩 余 率 : ".sprintf("%%-01.2f",((1-$rightarr['content"]/2000000) | 
*]100))."%</p>"; | 

29 } 

30 } 

31 ?> 


3.4.3 ”为 用 户 划分 个 人 文件 目录 


为 注册 的 每 一 个 用 户 创建 一 个 与 其 注册 名 同名 的 个 人 文件 目录 ,这 样 不 但 可 以 方便 管 | 
理 员 对 个 人 文件 的 管理 ， 同 时 也 避免 了 所 有 用 户 上 传 文件 到 同一 文件 目录 下 的 浆 端 。 | 

创建 个 人 文件 目录 的 操作 在 用 户 的 注册 处 理 页 register_chk.php 中 完成 。 首先， 获取 表 | 
单 提交 的 用 户 注册 信息 ; 然后 ， 执 行 添加 语句 ， 将 注册 信息 添加 到 指定 的 数据 表 ， 最后， 
通过 mkdir0 函 数 在 服务 器 指定 的 目录 下 创建 一 个 与 用 户 名 相同 的 文件 夹 。 其 关键 代码 如 下 


oe Ll 


了 tp 各 并 扩 志 煤 种 


<?php 
session start(); 
header('Content-Type:text/html:charset=gb2312"); 
include_once 'conn/conn .php'; 
Sreback = "0'; 
$sql = "insert into tb_member(name,password,question,answer,email) 
values(".trim($ GET['name"]D).",".md5(trim($ GET['pwd"])).",".$_GET['question'].", 
™.$ GET['answer].",".$_ GET['email].")"; 
$num = $conne->uidRst($sq)); 
if($num = 一 1){ 
$_SESSION['name'] = $_GET['name']; 
Susername=$ SESSION[mame']: 
S$dimame="../upfile/$usermame"; // 设 置 文件 夹 的 路 径 及 名 称 
mkdir($dirname,0700); // 创 建 一 个 和 用 户 名 相同 的 文件 夹 
Sreback = "1'; 


3.4.4 ”防止 用 户 文件 目录 的 非法 访问 


为 了 避免 某 些 用 户 通过 不 正当 方式 获取 其 他 用 户 文件 夹 的 地 址 ， 从 而 直接 通过 地 址 访 
问 其 他 用 户 文件 夹 ， 获 得 相关 的 内 容 并 下 载 其 他 用 户 文件 。 可 在 为 每 个 用 户 创建 文件 夹 的 
同时 ， 在 个 人 文件 夹 下 创建 一 个 index.php 文件 ， 以 此 来 屏蔽 直接 通过 地 址 栏 访 问 文件 夹 


下 的 内 容 。 


首先 ， 在 login 文件 夹 下 创建 一 个 index.txt 文本 文件 。 代 码 如 下 : 


01 
02 
03 
04 
05 
06 


<?php 
echo "请 使 用 正确 的 访问 路 径 !"; 
?> 
<p> 
点 击 <a hre 舍 "./../index.php"> 这 里 </a> 返 回 /| 单 击 返回 首页 
</p> 


然后 ， 在 注册 信息 处 理 页 register_chk.php 中 ， 完 成 用 户 注册 信息 的 添加 之 后 ， 将 
index.txt 文本 文件 复制 到 新 创建 的 用 户 个 人 文件 夹 下 ， 并 且 命 名 为 index.php。 其 关键 代码 


如 下 : 


Sfile = "index.txt'; 
Snewfile="../upfile/$username/index.php"; 
if (!copy(Sfile, Snewfile)) { 

echo "failed to copy $file...\n"; 


// 设 置 复制 新 路 径 名 称 
// 将 index.txt 复制 到 新 路 径 


} 
jelse{ 
Sreback = $conne->msg_error(); 
echo $reback: 
?> 
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3.5 找 回 密码 


3.5.1 找 回 密码 概述 着 会 


长 时 间 未 登录 的 用 户 可 能 会 忘记 网 站 的 登录 密码 :另外 ， 一 些 用 户 的 密码 也 可 能 会 被 其 A29 
他 人 非法 更 改 。 为 此 ， 必 须 创建 一 个 密码 找 回 模块 ， 为 个 人 用 户 取 回 密 码 提供 一 个 有 效 平 | 
台 , 从 而 保证 每 一 位 用 户 能 够 及 时 找 回 登录 密码 。 密 码 找 回 模块 的 运行 效果 如 图 3.13 所 示 。 | 


ee ) 


| 


-Wi 


人 hetpy/127.0.0.UTM/03/login/found.php | hapy/12700.UVTM/O3yiegiyfound ,php 


图 3.13 密码 找 回 模块 的 运行 效果 


在 本 模块 中 ， 找 回 密码 采用 的 是 密 保 问题 与 密 保 答案 比较 的 方式 。 首 先 用 户 输入 注册 
的 用 户 名 ， 然 后 输入 密 保 问题 和 密 保 答案 ， 最 后 对 用 户 提交 的 问题 和 答案 与 数据 库 中 存储 | 
的 数据 进行 比较 ， 如 果 问 题 和 答案 都 正确 ， 则 该 用 户 可 以 执行 密码 的 修改 操作 ， 重 新 设置 
新 的 密码 。 


3.5.2 ” 找 回 密码 页 面 设 计 


在 找 回 密码 页 面 中 ， 关 键 是 创建 表单 ， 提 交 注 册 的 用 户 名 、 密 保 问题 和 密 保 答 案 ; 同 
时 还 嵌入 了 两 个 不 同 ID 值 的 div 层 , 通过 控制 div 层 样式 的 隐藏 和 展开 来 实现 密码 找 回 表 | 
单 和 更 改 密码 表单 之 间 的 切换 操作 。 找 回 密码 文件 found.php 的 关键 代码 如 下 : | 
01 <!-- 找 回 密码 部 分 --> 
02 <div id="foundinfo"> 
03 <table width="250" border="1" align="center" cellpadding="0" cellspacing="0"> 
04 <tr> 


05 <td height="25" colspan="2" align="center" valign="middle"> 密 码 找 回 </td> 
06 </t> 

07 <tr> 

08 <td width="100" height="25" align="right" valign="middle"> 找 回 账号 ， </td> 
09 <td height-"25">&nbsp: 


10 <input id="foundname" type="text" style=" width: 100px; height:15px:; border:lpx #000000 | 
solid:" /></td> | 


向 </tr> 
2 <!-- 省 略 部 分 代码 --> 

13 <tr> 

14 <td height="25" colspan="2" align="center" valign="middle">&nbsp; 


i Lu 
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27 
| 28 
| solid;" /></td> 
29 
30 
31 
32 
| 33 
| solid:" /></td> 
| 34 
35 
| 36 
| 37 
| 38 
| 39 
40 


| 在 found.php 文件 中 ， 载 入 了 js\found.js 脚本 文件 ， 通 过 Ajax 无 刷新 技术 调 上 


<button id="stepl" class="btn"> 下 一 步 </button></td> 
</tr> 
</table> 
</div> 
<!-- 更 改 密码 部 分 --> 
<div id="changepwd" style="display:none;"> 
<table width="250" border="1" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td height="25" colspan="2" align="center" valign="middle"> 更 改 密码 </td> 
</tr> 
<tr> 
<td width="100" height="25" align="right" valign="middle"> 输 入 密码 : </td> 
<td height="25">&nbsp:; 
<input id="pwdl" type="password" style=" width: 100px: height:1Spx: border:1px #000000 


</tr> 
<tr> 
<td width="100" height="25" align="right" valign="middle"> 二 次 密码 : </td> 
<td height="25">&nbsp:; 
<input id="pwd2" type="password" style=" width: 100px: height:1Spx: border:1px #000000 


</tr> 
<tr> 
<td height="25" colspan="2" align="center" valign="middle">&nbsp: 
<button id="step2" class="btn"> 完 成 </button></td> 
</tr> 
</table> 
</div> 


| found_chk.php 文件 , 对 用 户 提 交 的 密 保 问题 和 密 保 答案 进行 验证 , 并 且 根 据 found_chk.php 
文件 中 的 返回 值 控制 div 标签 的 隐藏 和 显示 。foundjjs 脚本 文件 的 代码 如 下 : 


| ol 


S$('step1").onclick = functionO{ 
| 02 if($('foundname').value !=" && $('question").value !=" && $('answer'").value (= "){ 
| 03 xmlhttp.open('get','found_chk.php?act=sel&foundname="+$('foundname') 
| 04 .valuet'&question="+$('question').valuet'&answer='+$('answer').value,true); // 验 证 找 回 
| 密码 预 设 信息 
| 05 xmlhttp.onreadystatechange = function(|{ 
| 06 if(xmlhttp.readystate 一 4 && xmlhttp.status 一 200){ 
| 07 msg = xmlhttp.response Text: 
| 08 if(msg =='1){ 
| 09 S$(foundinfo').style.display = mone' /更 改 display 属 
性 隐藏 pundinfo 层 
| 10 S$('changepwd').style.display ="; // 更 改 display 属 
， 性 显示 changepwd 层 
| 11 }else{ 
| 12 alert(' 填 写 信息 错误 ! ); 
| 13 } 
| 14 } 
| 15 } 


。74 。 


第 3 章 上 伟 与 下 载 模块 


16 xmlhttp.send(nulD):; 
1 hs 
18 } ! 
3.5.3“” 找 回 密码 处 理 | 鳞 ” 


创建 找 回 密码 处 理 文件 found_chk php。 该 处 理 文件 主要 包括 两 个 不 同 的 处 理 方法 ， 即 
证 密码 保护 问题 和 更 改 密码 。 首 先 通 过 URL 获取 act 的 参数 值 来 判断 用 户 执行 的 操作 ， | 
然后 根据 选择 的 act 参数 来 执行 相应 的 方法 , 对 用 户 提交 的 密 保 问题 和 密 保 答案 进行 验证 ，| 
或 者 完成 对 用 户 提交 新 密码 的 更 新 操作 。 其 关键 代码 如 下 : 


01 <?php 

02 header('Content-Type:text/html:;charset=gb2312"); 

03 include_once 'conn/conn.php'; 

04 $act=$_GET['act]:; 

05 Sreback = '0'; 

06 if($act = 'sel){ 1/ 判断 用 户 选 择 的 操作 
07 Sname = $_GET['foundname']; /获取 找 回 密码 用 户 名 
08 $question =$_GET['question']; // 找 回 密码 问题 

09 $answer =$_GET['answer]: // 找 回 密码 


10 $sql="select * from tb member where name=".$name." and question="".$question." and | 
answer=".$answer.™"; | 


11 $num = $conne->getRowsNum($sq]); 

12 }else if($act == "up) { 

13 Sname = $_GET['foundname']; 

14 Spwd =$_GET['pwd']: 

15 $sql = "update tb_ member set password = ".md5($pwd)." where name =".$name."™"; 
16 Snum = $conne->uidRst($sql): 1/ 执行 数据 库 更 新 操作 
1 } 

18 if($num !="){ 

19 Sreback = "1'; // 返 回 值 为 1 

20 jelse{ 

21 Sreback = $sql: 

22 } 

23 echo $reback: 

24 


说 明 : 
用 户 提交 的 新 密码 首先 会 经 MD5(0) 函 数 加 密 ， 然 后 再 被 插入 数据 库 中 。 


3.6 文件 上 传 


3.6.1 文件 上 传 概述 


作为 网 络 移动 存储 工具 ， 上 传 是 必 不 可 少 的 功能 之 一 。 用 户 可 以 通过 登录 个 人 账户 ， | 
单 击 “ 上 传 文件 ” 超 链接 ， 进 入 文件 上 传 页 ， 将 自己 想 要 上 传 的 工具 、 文 档 、 音 频 或 视频 | 


L_ 


.App 内 晶 搞 半 精 角 


等 多 种 类 型 的 文件 上 传 到 个 人 空间 中 。 文 件 上 传 页 面 的 运行 效果 如 图 3.14 所 示 。 


Note 


3.6.2 文件 上 传 表单 设计 


MR 甸 E 网 络 u 熏 

RR a 向 首页 | 用 加 At 天 严 | 六 设 为 首页 

项 通 和 本: me 
上 传 文件 | 下 载 文 六 

侈 是 [过 请 放 己 ] 明日 网 络 U 盘 
个 再 杂 宇 则 ;2.000B 

简 这 这 ;100.00% 

i Ta:| EE :I [可 习 问 
屠 上 传 文件 
Lf :1000 
C:\Documents ad settire Nae [TAH ld] 
LL | 


图 3.14 文件 上 传 页 面 的 运行 效果 


| 文件 上 传 要 通过 表单 中 的 文件 域 来 提交 ， 在 处 理 页 中 通过 $_FILES[] 全 局 变量 和 
| Imove_upload file(O) 函 数 完 成 文件 的 上 传 操作 。 

| 上 传 文件 的 表单 存储 于 upfile.php 文件 中 。 首 先 ， 创 建 form 标记 ， 指 定 POST( 方 法 
| 提交 ， 设 置 enctype 属性 等 于 "multipart/form-data"， 设 置 处 理 页 upfile_chk.php。 然 后 ， 创 
建文 件 域 ， 选 择 上 传 文件 。 接 着 ,创建 下 拉 列 表 框 展示 文件 类 型 ， 将 从 数据 库 读 取 的 文件 
类 型 作为 下 拉 列 表 框 的 值 。 接 下 来 ,创建 下 拉 列 表 框 ， 设 置 文件 权限 。 最 后 ， 创 建 提交 按 
钮 ， 通 过 onClick0 事 件 调 用 JavaScript 脚本 中 的 方法 验证 提交 数据 ， 完 成 提交 操作 。 
upfile.php 文件 的 关键 代码 如 下 : 


01 
02 


<!-- 上 传 表单 ， 处 理 页 为 upfile_chkphp--> 

<form method="post" enctype="multipart/form-data" action="?act=upfile"> 

<!-- 列 表 项 --> 

<ul id="change"> 

<!-- 上 传 表单 。 注 意 : name 值 为 一 个 数组 --> 

<liid="addordel"> 

<input id="upname[]" name="upname[]"” type="file" style=" width: 200px; 


background-color:#f0f0f0; " />&nbsp:; 


09 
10 
11 


<!-- 文 件 类 别 下 拉 列 表 ，name 也 是 数组 --> 
<select id="foundtype[]" name="foundtype[]"> 


<2php 
$typesql = "select genrename from tb_uptype"; /查询 语句 
Stypearr = $conne->getRowsArray($typesq]): // 返 回 查询 结果 集 
$conne->close_rst(); // 清 空 数组 
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15 foreach($typearr as $value){ /循环 输出 类 别 列 表 ! 
1690 3 | 
机 <option value="<?php echo S$value['genrename']; ?>"><?php echo S$value | 


['genrename']:; ?></option> | 


18 <?php | 5 
19 } | 食 - 


20 ?> 

21 </select>&nbsp; 8 
22 ”<!-- 判 断 文 件 是 否 公开 。0， 不 公开 ; 1， 公 开 --> | 
23 <select id="ispub[]" name="ispub[]"> | 
24 <option value="0" selected="selected"> 不 公开 </option> 
25 <option value="1"> 公 开 </option> 

26 </select>&nbsp; | 
27 <!-- 删 除 上 传 域 和 添加 上 传 域 -> 
28 </li> | 
29 </u> 

30 <input id="upbtn" name="upbtn" type="submit" value=" 上 传 " 

31 onClick="return chkfile()" style=" background-color:#f0f0f0; border: 1px #CCCCCC solid:" /> 
32 </form> 


3.6.3 ”文件 上 传 处 理 


创建 文件 上 传 处 理 页 upfile_chk.php。 首 先 ， 获 取 表单 提交 的 数据 ， 定 义 文件 的 存储 路 
径 。 然后 ,判断 是 否 存 在 重 名 文件 ,判断 上 传 文件 大 小 是 否 符合 要 求 ， 生 成 下 载 码 。 接 着 ， 
完成 文件 的 上 传 操作 ， 并 且 将 文件 信息 、 下 载 码 和 空间 使 用 信息 存储 到 数据 库 中 。 最 后 ， 
输出 下 载 码 ， 返 回 上 一 页 。 其 关键 代码 如 下 : 


01 include_once 'conn/conn.php'; 


02 S$filename = $ FILES[upname'][mame']; // 读 取 上 传 文件 名 ， 并 保存 为 数组 

03 Sfiletype = $_POST['foundtype']; // 读 取 上 传 文件 的 类 别 

04 Stmpname 二 $_FILES["upname'][tmp_name']; 。“”// 读 取 临 时 文件 名 ， 并 保存 为 数组 

05 Stmpsize = $_FILES["upname']['size']: /上 传 文件 大 小 

06 S$tmppub =$ POST[ispub']; // 是 否 公开 

07 $dimame=$ SESSION[mame']:; 1/ 选取 和 用 户 登录 名 相同 的 个 人 文件 夹 ! 
08 Sfile_path = "upfile/$dirmame/"; // 文 件 保存 路 径 | 
09 Smax = 0; /文件 大 小 ， 初 始 化 为 0 | 
10 for($j=0:$j<count($_ FILES[mpname'][mame]):Sj++) { ”// 判 断 上 传 的 文件 是 否 已 存在 | 
11 Sfilearr=$_FILES[‘upname'][‘name'][$j]: 1/ 获取 上 传 的 文件 名 | 
12 S$filenamei=$file path.$filearr: /设置 上 传 文件 路 径 | 
13 if(file_exists($filenamei)) // 判 断 同 目录 下 文件 是 否 存在 | 
14 $construct=0; | 
15 else | 
16 $construct=1; // 如 果 存 在 则 返回 0, 不 存在 则 返回 1 | 
17 } | 
18 if($construct!=1) { // 判 断 文件 存在 ， 则 提示 文件 名 冲突 | 


19 ”echo '<script>alert(" 文 件 名 称 冲突 ! 请 重新 命名 后 上 传 !"); | 
20 location=".$ SERVER['HTTP REFERER'.":</script>"; 
21 由 
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| RN 
| 22 iflis array(Stmpsize)){ // 如 果 变 量 $tmpsize 不 是 数组 
| 23 foreach($tmpsize as $value){ 1/ 获取 文件 总 的 大 小 
| 24 Smax += $value: 
| 25 
26 这 $max > 100000000 or Smax <= 0){ 


27 ”echo '<scripf>alert(" 上 传 文件 总 大 小 大 于 100MB， 请 重新 选择 "); 
28 location=".$_ SERVER[IHTTP REFERE R'].";</script>"; 


58 Schkdownnum = "; 
59 for($i=0; $i<15: itt){ 


29 1 
| 30 Jelse{ // 重 新 选择 上 传 文件 
| 31 echo '<script>alert(" 上 传 文件 错误 ， 请 重新 检查 程序 "); 
| 32 location=".$ SERVER['HTTP REFERER'].":;</script>'; 
| 23 
| 34 $rightsql = "select * from tb_ member where name =".$ SESSION['name'].""; 
| 35 Srightarr = $conne->getRowsRst($rightsq]): 
| 36 $conne->close_rst(|; 
| 37 switch($rightarr['right']) { 1/ 判断 用 户 权限 
| 38 case '0': 
| 39 S$conmax=2000000: // 按 当前 登录 用 户 的 权限 分 配 的 最 大 使 用 空间 
| 40 break: 
| 41 case 'l': 
| 42 Sconmax=3000000; 
| 43 break 
| 44 case'2': 
| 45 $conmax=5000000; 
| 46 } 
| 47 echo "<p>” 
| 48 Slesarr=$conmax-$rightar['content]; /将 该 用 户 剩余 空间 大 小 赋值 给 lesarr 变量 
| 49 if($conmax-$rightarr['content] <= 1000000) { /如果 个 人 剩余 空间 已 经 不 足 1MB 
| 50 echo '<script>alert(" 您 的 上 传 下 载 空间 不 足 ， 请 清理 后 上 传 文件 "); 
| 51 location=".$ SERVER['HTTP REFERER'].";</script>'; // 提 示 空 间 不 足 
| 52 } 
| 53 else if($lesarr-$max<=1000000) // 如 果 上 传 文件 大 小 超过 剩余 空间 大 小 
| 54 上. 
| 55 echo '<script>alert(" 您 的 上 传 下 载 空 间 不 足 ， 请 整理 后 上 传 文件 "); 
| 56 location=".$ SERVER['HTTP REFERER'.";</script>'; // 提 示 空 间 不 足 
| 57 } 
| 60 Schkdownnum .= dechex(rand(0,15)): 
| Su 
| 62 for($i=0:$i< count(Sfilename); $i++){ // 通 过 循环 将 多 个 文件 全 部 上 传 
| 63 move_uploaded file($tmpname[$i].$file— path.$filename[$i]): 
| 64 // 添 加 数据 库 
| 65 Sinsertsql = 'insert into tb_upfile (filename,filepath,filetype,upauthor, chkdownnum, ispub, filesize) 
| 66 values(".trim($filename[$i]).",".$file_path.$filename[$i].",".$filetype[$1].", 
| 67 "S$ SESSION[name].",".trim($chkdownnum).",'.$tmppub[$i].,.$tmpsize[$i].)'; 
| 68 $conne->uidRst(Sinsertsql): 
6 
70 $maxtotal=S$rightarr['content']+$max:; // 从 数据 库 得 到 该 用 户 已 使 用 空间 大 小 与 新 上 
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传 文件 占用 空间 大 小 的 总 和 
71 /更 新 数据 库 
72 $csql="update tb_member set content = $maxtotal where name= ".$_ SESSION[mame'].""; 
73 Sconne->uidRst($csql); | 
74 ”echo '<div style=" padding-top: 30px:"> 文 件 上 传 成 功 ， 可 以 通过 下 载 码 : 贪 凡 


75 <font color="#FF0000">'.$Schkdownnum.'</font> &nbsp: 对 文件 进行 下 载 。</div>'; 
76 ”echo "<p> 点 <a href-\index.php?act-up\><span style=\"color:red\" > 这 里 <lspan></a> 继 续 上 传 1 MNM 
其 他 文件 ! </p>"; ote 


说 明 : | 
下 载 码 是 一 组 16 位 长 的 十 六 进 制 字符 囊 。 先 由 Tand() 函 数 生成 一 个 0 一 15 之 间 的 随机 | 
数 ， 再 使 用 dechex() 函 数 将 该 数 转换 为 十 六 进 制 表 示 ， 最 后 使 用 for 循环 重复 16 次 。 如 果 | 
一 次 上 传 多 个 文件 ， 那 么 所 有 文件 的 注册 码 为 同一 个 。 | 


37 文件 下 载 


3.7.1 功能 概述 


文件 下 载 是 上 传 、 下 载 模块 的 另 一 核心 内 容 。 首 先 ， 无 论 用 户 登 录 与 否 ， 都 可 以 对 已 | 
公开 的 文件 进行 下 载 ， 当 用 户 登 录 后 ， 不 但 可 以 上 传 新 的 文件 ， 也 可 以 对 已 经 上 传 的 文件 | 
执行 下 载 和 删除 操作 ;如 果 当 前 没有 登录 ， 作 为 游客 身份 ， 也 可 以 通过 获得 与 文件 匹配 的 | 
下 载 码 ， 对 不 公开 文件 执行 下 载 操作 。 文 件 下 载 页 面 的 运行 效果 如 图 3.15 所 示 。 | 


1A | ne 


图 3.15 文件 下 载 页 面 的 运行 效果 
3.7.2 通过 下 载 码 下 载 文 件 | 

用 户 登录 后 ， 可 以 通过 单 击 “ 下 载 文件 ” 超 链接 或 输入 下 载 码 对 文件 进行 下 载 。 这 里 | 
以 下 载 码 下 载 为 例 ， 讲 解 文件 下 载 的 实现 过 程 。 | 


下 载 码 下 载 的 操作 在 index.php 文件 中 完成 。 首 先 ， 创 建 下 载 码 文本 框 ， 提 交 下载 码 。 
其 关键 代码 如 下 : | 


oe (os L_ 


Note 


.pp 训 蜡 要 志 煤 角 


<div 1d="found">&nbsp; 

下 载 码 : 

<input id="chkdown" type="text" />&nbsp; 

<button id="downbtn" onclick="chkdownO"></button> 


在 index.php 文件 中 , 当 用 户 单 击 “ 下载 ”按钮 时 ,将 触发 onClick0 事 件 , 调用 chkdown() 


函数 判断 输入 值 是 否 为 空 ， 如 果 不 为 空 ， 将 重新 加 载 本 页 ， 并 将 下 载 码 作 为 参数 传递 


chkdownO 函 数 存储 于 js\indexjs 脚本 文件 中 ， 其 关键 代码 如 下 。 


function chkdownO{ 
if($(‘chkdown'").value =— "){ 
alert(' 请 输入 下 载 码 !"); 
return false; 
} 
location = ?act=downcode&chkdown='+$('chkdown').value; 
| 


然后 ， 在 index.php 页 中 ， 当 超 链接 传递 的 值 等 于 downcode 时 ， 加 载 downcode.php 


”页 。 其 关键 代码 如 下 : 
| 01 Pe 
02 ”<!-- 根 据 不 同 的 参数 值 ， 载 入 不 同 的 页 面 --> 
03 <div id="content"> 
04 <?php 
05 $act=$ GET['act]: 
06 switch($act){ 
07 case up': 
08 include upfile .php': // 载 入 上 传 页 
09 break; 
10 case "upfile': 
11 include "upfile_chk.php'; // 载 入 上 传 处 理 页 
12 break; 
13 case 'down': 
14 include ‘downfile.php'; // 载 入 下 载 页 
15 break: 
16 case 'downcode': 
17 include ‘downcode.php'; // 载 入 下 载 码 下 载 页 
18 break; 
19 default: 
20 include ,pub.php': // 载 入 共享 文件 页 
| break:; 
2 } 
23 ?> 
24 </div> 
| 接着 ， 创 建 downcode.php 页 ， 根据 获取 的 下 载 码 生成 SQL 查询 语句 ， 循 环 输出 下 
载 文件 信息 ; 并 且 创 建文 件 下 载 超 链接 ， 链 接 到 download.php 文件 ， 完 成 文件 下 载 操 
| 作 。 其 关键 代码 如 下 : 
| 01 <?php 
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02 $chkdown =tnm($_ GET['chkdown']): 
03 $downsql = "select * from tb_ upfile where chkdownnum = ".$chkdown."™"; 
05 ?> 


06 ”<!-- 下 载 文件 列表 --> 


04 $downarr = $conne->getRowsArray($downsq]): | 
07 <divid="filelist"> | 


08 -<?php 
09 S$infoline = 0; 8 
10 ”这 !empty($downarD){ /如 果 结 果 集 数组 不 为 空 | 
11 ?> | 
12 <ul> | 
13 <li style=" width: 30px; background-color: #F8E7AS:">&nbsp;</li> | 
14 <li style=" width: 200px; background-color:#F8E7A5; "> 文件 名 </li> | 
15 <li style=" width: 50px; background-color:#F8E7A5; "> 文件 类 型 </li> | 
16 <li style=" width: 100px; background-color:#F8E7A5;"> 下 载 码 </li> | 
17 <li style=" width: 150px; background-color:#F8E7A5;"> 上 传 时 间 </li> | 
18 </u> | 
19 ”<?php | 
20 foreach($downarr as $key => $value){ /循环 输出 数组 | 
21 ?> ! 
2 <ul> | 
23 <li style=" width: 30px:background-color: <?php echo ($infoline %2 一 0?#FFFFFF': | 
#f0f0f0"); ?>;"> | 


24 <input id="chk[<?php echo $key; ?>]" type="checkbox" value="<?php echo $value['id']; ?>" 

25 style=" width:30px; height:20px: border: 0px:" /></li> 

26 <li style=" width: 200px; background-color:<?php echo ($infoline %2 
==07'#FFFFFF':'#f0f0f0"); ?>:"> 

2 Rnbsp; <a href="download.php?id=<?php echo S$value["id]; ?>"><?php echo 
$value['filename']; ?></a></li> 

28 <li style=" width: S0px: background-color: <?php echo ($infoline %2 | 
==07'#FFFFFF':'#f0f0f0"); ?>:"> | 

29 &nbsp; <?php echo $value['filetype']; ?></li> | 

30 <li style=" width: 100px: background-color: <?php echo ($infoline %2 | 
=—07#FFFFFF'-#O{0f0"); ?>;"> 

31 &nbsp; <?php echo $value['chkdownnum']; ?></li> 

也 <li style=" width: 150px: background-color: <?php echo ($infoline %2 
==07?#FFFFFF"':#f0f0f0"): ?>:"> 

33 &nbsp; <?php echo $value[uptime']: ?></li> 


34 </ul> 

35 <?php 

36 S$infoline = ($infoline+1)%2; 
37 } 

38 }else{ 

39 echo ' 没 有 可 下 载 资源 '; 

40 上 

41 > 

42 </div> 


最 后 ， 创 建 download.php 文件 ， 根 据 文件 的 完整 路 径 和 headerO 函 数 完成 下 载 。 其 关 
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ow 
ER 
Wc 


| 键 代码 如 下 ， 


01 <?php 


.pp 凑 和 村 半 精 角 


02 include_once 'conn/conn .php': 


03 
04 
05 
06 
07 


18 ?> 


$id=$ GET[id]: /获取 用 户 的 id 

$downsql = "select* from tb_upfile where id = $id"; /通过 where 子 句 获得 相应 的 文件 
Sdownarr = $conne->getRowsRst($downsql): 

$path=$downarr['filepath']:; // 获 取 路 径 名 称 赋 给 $path 


if(!empty($path) and !is_null($path)){ 
S$filename=basename($path); 
$file=fopen($path,"r"); 
header("Content-type:application/octet-stream"); 
header("Accept-ranges:bytes"); 
header("Accept-length:".filesize($path)): 
header("Content-Disposition:attachment:filename=". $filename); 
echo fread($file,filesize($path)):; 
felose($file): 
exit; 


3.7.3 同步 删除 文件 以 及 释放 用 户 空间 


| 单 击 “下 载 文 件 ” 超 链接 后 ， 会 在 下 载 文件 的 下 方 看 到 “删除 选择 的 文件 ” 超 链 接 ， 

| 单 击 该 超 链接 , 在 删除 数据 库 文件 的 同时 ,也 会 删除 该 登录 用 户 相 应 文件 夹 下 的 同名 文件 ， 
| 并 且 同 步 更 新 用 户 使 用 空间 的 大 小 ， 达 到 删除 文件 释放 空间 的 效果 。 删 除 文件 1 秒 钟 之 后 
| 将 返回 下 载 首 页 。 删 除 文件 释放 空间 的 操作 是 在 downfile php 文件 中 完成 的 ， 其 关键 代码 


| 如 下 : 
| 01 <?php 

02 $del = $_GET['del"]; 

03 Snum =10: // 每 页 显示 记录 数 

04 Scurpage = $_GET['curpage']; // 当 前 页 

05 $sql = "select * from tb_upfile where upauthor = ".$_SESSION['name'].""; 

06 Stotnum = $conne->getRowsNum($sq]); // 记 录 数 

07 if($del — 'del){ 

08 Srd=S$_GET[ad]: 

09 $delsql = "delete from tb_upfile where id = -1 "; 

10 S$deltotal = "select sum(filesize) as total fom tb_ upfile whereid=-1" /取得 

所 选 文件 的 总 大 小 

11 Scontent="select * from tb_member where name =".$9 SESSION[mname']."": 

12 Scontarr = $conne->getRowsRst($content): 

13 if(!empty($rd){ 
| 14 S$delarr = explode(',".$rd); 

15 foreach($delar as $value){ 
| 16 /以 下 为 新 增 内 容 ， 在 删除 文件 的 同时 也 可 以 删除 硬盘 文件 夹 上 的 文件 
| 17 $sql= "select * from tb_upfile where id = ".$value." "; /选取 
， 删除 文 件 的 过 值 


J 
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18 $delsql .=" orid = ".$value."": | 

19 Sdeltotal =" orid=".Svalue." "; | 

20 S$sqlarr = $conne->getfRowsRst($sq]); // 调 用 SQL 函数 | 

2 $delpath=$sqlarr['filepath']; // 绪 得 删除 文件 路 径 | 

22 unlink(Sdelpath): 1/ 删除 指定 路 径 的 文件 | 傣 六 

23 } .A 

24 Sdeltotalnum=$conne->getRowsRst($deltotal); | NM 

25 Snewdata = $contarr['content]-$deltotalnum['total']; a ote 

26 // Snewdata 为 删除 文件 后 个 人 空间 的 使 用 大 小 | 

27 Snewsql="update tb member set content = $newdata where name = | 
m.$_SESSION[mame]."": | 

28 S$conne->uidRst(Snewsq]); // 把 删除 文件 后 剩余 空间 大 小 更 新 到 数据 库 “| 

29 $delnum = $conne->uidRst($delsql); | 

30 echo "<meta http-equiv=\"refresh\" content=\"1;url=index.php?act=down\">"; //1 | 
秒 钟 之 后 跳 转 主页 | 

31 echo "<p></p>"; | 

32 echo "删除 成 功 ! "; 

33 } 


3.8.1 后 台 管 理 概述 


在 后 台 管理 系统 中 只 有 一 个 功能 , 就 是 管理 注册 用 户 的 权限 。 其 运行 效果 如 图 3.16 所 示 。 | 
用 户 权 限 管理 


目前 共有 2 条 记录 共 分 1 页 显示 当前 显示 第 ! 页 


用 户 名 ”使 用 权限 。” 使 用 容量 。 修改 
mrsoft ”高 级 用 户 0.00MB 
leonsk ”高 级 用 户 0. 00NB 蜂 


第 一 页 | 上 一 页 | 下 一 页 | 最 后 一 页 


图 3.16 用 户 权 限 管理 

3.8.2 ”用 户 权 限 管理 页 面 设计 | 
进入 到 后 台 管理 系统 中 ， 以 分 页 形式 输出 已 经 注册 的 用 户 信息 ， 包 括 用 户 名 、 使 用 权 | 
限 、 使 用 容量 ， 以 及 “修改 ” 超 链接 。admin.php 文件 的 关键 代码 如 下 : | 
01 <?php | 


02 include_once 'conn/conn.php': 
03 Slist_ num=10; ! 


04 $sql = "select * from tb _ member": // 查 询 语句 | 
05 $sqlarr = $conne->getRowsRst($sq]): // 返 回 查询 结果 集 | 
06 S$totnum = $conne->getRowsNum($sq]); // 获 取 查 询 记录 数 | 


07 $conne->close rstO: 
08 if($totnum>0){ 


。83 。 


.pp 内 晶 搞 半 精 角 


<p> 
<table> 

<tr class="trl"> 
<td> 用 户 名 </td> 
<td> 使 用 权限 </td> 
<td> 使 用 容量 </td> 
<td> 修 改 </td> 
</tr> 

<?php 


$sql = $sql." order by id desc limit ".($list_ num *($page-1)).",".$list_num; 


Sfilearray = $conne->getRowsArray($sq]): 
foreach($filearray as $value){ 
echo "<tr>\n"; 
echo "<td>".$value['name']."</td>\n"; 
Switch($value['right']){ 
case 'O': 
echo "<td> 普 通用 户 </td>\n"; 
break: 
case ']': 
echo "<td> 会 员 用 户 </td>\n"; 
break:; 
Case 2': 
echo "<td> 高 级 用 户 </td>\n"; 
break: 


} 


// 查 询 语句 


// 查 看 用 户 权限 


echo "<td>".sprintf("%-01.2f",($value['content"]/1000000))."MB</td>\n"; 
echo "<td><a href=changesub.php?id=".$value["id1."> 修 改 </a></td>\n"; /获取 这 更 


echo "</tr>\n"; 


} 
echo "</table>"; 


3.8.3 ”更改 用 户 权限 表单 设计 


件 的 关键 代码 如 下 : 


01 
02 


<table border=1> 

<form action="change.php" method="post"> 
<?php 

include_once ‘conn/conn.php'; 

$sql="select * from tb_member where id = $id"; 
$changearr = $conne->getRowsRst($sq]): 

3 

<tr> 
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并 且 将 该 信息 在 表单 中 
户 权 限 的 修改 操作 。changesub.php 文 


更 改 用 户 权限 , 是 从 用 户 权限 管理 页 中 设置 的 “修改 " 超 链接 开始 的 。 在 changesub.php 
文件 中 ， 根 据 超 链接 传递 的 ID 值 ， 查 询 出 指定 用 户 的 详细 信息 ， 
输出 ， 进 而 通过 表单 将 数据 提交 到 指定 页 ， 完 成 对 


/查询 语句 
/返回 查询 结果 
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09 ”<td> 用 户 名 </td> 
10 <td><?php echo $changearr['name']:?></td> 


11 </e> | 
IE | 

13 ”<td> 当 前 权限 </td> | 
14 <td> | 食 - 
15 <2 

16 下 // 获 取 用 户 权限 ”Note 
17 case 'O': | 

18 echo "普通 用 户 "; | 

19 break; | 

20 case ']': | 

21 echo "会 员 用 户 “ | 

3 break:; | 

23 Case 2': | 

24 echo "高 级 用 户 "; 

5 break: 

26 } 

27 > 

28 </td> 

29 </tr> 

30 <tr> 

31 <td> 修改 权限 </td> 

32 <td> 

33 <selectname="right"> // 选 择 用 户 权限 


34 <option value="0" selected="selected"> 普 通用 户 </option> 

35 <option value="1"> 会 员 用 户 </option> 

36 <option value="2"> 高 级 用 户 </option> 

Ek </select> 

38 </td> 

3 </tr> 

40 <tr><td colspan=2> 

41 <center> 

42 <input type="submit" value=" 提 交 " /> 

43 </center> ! 

44 </td> 

45 </tr> 

46 <inputtype="hidden" name="id" value="<?php echo $changearr["id']?>" /> // 添 加 一 个 隐 | 
藏 域 传递 id 参数 
47 </form> 
48 </table> | 


3.8.4 执行 权限 更 改 操作 | 
在 changesub.php 文件 中 创建 指定 的 权限 更 改 处 理 文件 changephp, 获取 表单 中 提交 的 | 
数据 ， 完 成 用 户 权 限 的 更 改 操作 。change.php 文件 的 关键 代码 如 下 : | 


01 <?php | 
02 include_once 'conn/conn.php'; | 
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Sright=$ POST[right]: // 获 得 权限 更 改 表 单 的 传递 值 
$id=$_ POST[id]; 
Supsql = "update tb_member set tb_memberright =$right where id = $id"; // 查 询 语句 
$num=$conne->uidRst($upsq)); // 返 回 结果 
Sconne->close_rst(); 
echo "<p>"; 
if($num——=1){ 
echo "权限 更 改 成 功 !"; 
echo "<p>"; 
jelse{ 
echo "权限 更 改 失败 ! "; 
echo "<p>"; 
人 
<!-- 以 下 为 html 内 容 ， 修 改 成 功 都 会 返回 修改 主页 --!> 
<meta http-equiv="refresh" content="2;url=admin.php" /> 
2seconds later to return the manger page! 


3.9 技术 提炼 


”3.9.1 php.ini 配置 文件 


如 果 要 在 PHP 中 实现 小 文件 的 上 传 (2MB 以 下 )， 那 么 无 须 对 php.ini 配置 文件 进行 修改 ， 
| 使 用 默认 参数 即 可 。 但 如 果 想 实现 完美 的 上 传 功能 ， 则 一 定 要 对 配置 文件 有 所 了 解 。 在 php ini 


文件 中 ， 与 文件 上 传 有 关 的 设置 如 图 3.17 所 示 。 


lz44 ; Maximum execution time of each script, in seconds 


|: Maximum size of pos 下 data that PHP will accept. 
le1z post_max_size = 168 


sof Maximm alioved size for] uploaded files. 
ls17 |upload wax filesize = 32¥ 


ls1e 


图 3.17 php.ini 文件 中 的 相关 参数 
从 图 3.17 中 可 以 看 到 ， 有 如 下 3 部 分 内 容 需 要 进行 配置 。 
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1. Resource Limits | 


Resource Limits， 直 译 过 来 就 是 资源 限制 ， 其 中 包含 3 个 参数 。 该 区 块 不 仅仅 是 针对 
上 传 、 下 载 的 ， 而 是 对 全 部 的 文件 进行 设置 。 各 个 参数 含义 及 参数 值 说 明 如 表 3.1 所 示 。 | 
表 3.1 Resource Limits 块 的 参数 说 明 | 食 生 


Pr rn 
max_execution_time “| 每 个 脚本 页 面 完成 执行 的 最 大 时 间 ， 单 位 是 秒 。 如 果 设 为 -1， 说 明 没有 限制 _ 
max_input_time 每 个 脚本 页 面 处 理 请 求 数据 的 最 大 时 间 ， 单 位 是 秒 ， 也 可 以 设 为 -1 | 
memory_limit 一 个 脚本 页 所 能 够 消耗 的 最 大 内 存 


2. post_ max_ size 


post_max_size 参数 指 PHP 通过 表单 POST 所 能 接收 的 最 大 值 ， 包 括 表 单 里 的 所 有 项 。 


3. File Uploads 


File Uploads 块 是 专 为 文件 上 传 设置 的 。 该 区 块 中 同样 包含 了 3 个 参数 ， 各 个 参数 含义 
及 参数 值 说 明 如 表 3.2 所 示 。 | 
表 3.2 File Uploads 块 的 参数 说 明 


file uploads 是 否 允 许 HTTP 上 传 ， 默 认为 On， 即 开启 ， 无 须 修 改 


upload _ tmp _dir 文件 上 传 时 的 临时 存储 目录 。 如 果 没 指定 ， 将 会 用 系统 默认 的 临时 文件 夹 
upload_max filesize 允许 上 传 的 文件 最 大 值 


3.9.2 上传 表单 设置 


配置 文件 修改 完毕 后 ， 重 启 Apache 服务 器 。 接 下 来 ， 开 始 设计 <form> 表 单 。<form> | 
表单 可 以 实现 人 机 交互 ， 是 动态 网 页 的 一 个 最 基本 功能 。 最 初 的 <form> 表 单 并 不 支持 文件 | 
上 传 ， 表单 中 input 元 素 的 type 属性 只 有 7 种 值 ， 即 文本 框 〈text)、 密 码 框 (text)、 单 选 | 
框 (radio)、 多 选 框 (checkbox)、 重 置 按钮 (reset)、 确 定 按钮 (submit) 和 图 像 (image)。 | 
后 来 , rfe1867 为 HTTP 协议 添加 了 这 个 功能 ， 目 前 几乎 所 有 的 浏览 器 和 服务 器 都 支持 此 协议 。 | 
rfe1867 协议 为 type 属性 增加 了 file 值 ， 同 时 ，<form> 标 签 中 的 ENCTYPE 属性 值 必 | 
须 设 为 multipart/form-data，method 属性 则 必须 是 POST，3 个 条 件 缺 一 不 可 。 当 表单 属性 | 
设置 正确 后 ， 才 可 以 在 表单 处 理 页 中 获取 到 上 传 文件 的 值 。 下 面 是 简单 、 可 用 的 上 传 表单 | 
代码 : | 
<form method="post" action="2.php" enctype="multipart/form-data"> 

<input name="filename" type="file" /> 

<input type="submit" value="enter"/> 
</form> | 
此 外 ， 还 建议 在 上 传 表单 中 增加 一 个 hidden 隐藏 域 ， 隐 藏 域 的 name 值 为 | 
MAX FILE_SIZE，value 值 是 允许 上 传 的 最 大 字 节 数 。 例 如 : | 
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注意 : 


.pp 内 晶 搞 半 精 角 


CW 
ES 这 
SI 
<input type="hidden" name="MAX FILE SIZE" value="1024000" /> 

这 样 ， 当 上 传 文件 大 于 1MB 时 ，$_FILES 中 的 error 值 将 返回 2。 关 于 $_FILES 变量 


的 使 用 在 3.9.3 节 中 给 出 。 该 隐藏 域 的 作用 不 是 真正 要 去 限制 上 传 文件 的 大 小 , 因为 要 绕 过 该 
| 项 检查 非常 简单 ， 其 真正 作用 是 为 了 避免 用 户 误 传 了 大 文件 而 陷入 “无 尽 地 ”等 待 之 中 。 


通过 隐藏 域 控制 上 传 文件 的 大 小 ， 必 须 将 隐藏 域 放置 在 文件 域 之 前 ， 否 则 隐藏 域 不 会 


生效 . 
3.9.3 预定 义 变量 $_FILES 


如 果 表 单 设计 得 没有 问题 ， 那 么 在 处 理 页 中 ， 使 用 $_FILES 预定 义 变量 就 可 以 得 到 上 


传 文件 的 相关 信息 ; 反之 ， 如 果 得 不 到 ， 那 么 就 有 可 能 是 配置 文件 或 表单 设置 的 问题 了 。 
| 下 面 先 介绍 预定 义 变量 $_FILES。 


$_FILES 变量 存储 的 是 上 传 文件 的 相关 信息 ， 这 些 信 息 对 于 上 传 功 能 有 很 大 作用 。 该 


| 变量 是 一 个 二 维 数组 ， 保 存 的 信息 如 表 3.3 所 示 。 


表 3.3 $_FILES 变量 值 


参数 描 ” 述 
$_FILES[filename]['name" 存储 了 上 传 文件 的 文件 名 ， 如 exam.txt、myDream.jpg 等 
$_FILES[filename]['size" 存储 了 文件 大 小 ， 单 位 为 字 节 
FE 上传 果 首先 在 临时 录 中 被 | -个 临 采 件 。 该 变量 为 
$_FILES[filename]['tmp_name'] 和 上 传 时 ， 首 先 在 临时 目录 中 被 保存 成 一 个 临时 文件 。 该 变量 为 文 
$_ FILES[filename]["tmp_name'] | 上 传 文件 的 类 型 。 注 意 是 类 型 ， 不 是 后 级 
$_FILES[filenamel]['type'] 存储 了 上 传 文件 的 错误 代码 


error 字段 将 返回 上 传 文件 的 错误 代码 ， 一 共有 7 种 ， 具 体 说 明 如 表 3.4 所 示 。 
表 3.4 S$_FILES[filename][eror] 中 的 错误 代码 及 说 明 

错误 代码 错误 说 明 
UPLOAD ERR OK: 上 传 成 功 ， 无 错误 
UPLOAD_ERR_INIL SIZE: 文件 大 小 超出 php.ini 中 的 upload_max_filesize 的 参数 值 
UPLOAD ERR FORM _SIZE: 文件 大 小 超出 表单 中 MAX _FILE_SIZE 隐 藏 域 的 值 
UPLOAD ERR PARTIAL: 文件 没有 被 全 部 上 传 
UPLOAD ERR_NO FILE: 没有 文件 被 上 传 
UPLOAD ERR_NO_TMP_DIR: 找 不 到 临时 文件 目录 
UPLOAD ERR CANT WRITE: 文件 写 入 失败 


下 面 使 用 3.6.2 节 中 的 表单 上 传 一 个 文件 ， 使 用 $_FILES 变量 输出 文件 信息 。 代 码 如 下 : 


0 
1 
2 
3 
4 
6 
7 


01 <?php 

02 $tmpar = $_FILES['filename']; 1/ 获取 文件 域 的 值 

03 if($tmpar['name'] (= "){ /如 果 上 传 文件 名 不 为 空 
04 foreach($tmparr as $name => $value){ 
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05 echo $name.' => '.$value.'<br>"'; // 使 用 循环 输出 数组 索引 和 值 

06 } | 

07 Jelse{ ! 

08 echo "<script>alert(' 请 添加 上 传 文件 "):history.go(-1);</script>"; | 

09 } | 4 
1030 92> | 刁 认 


图 3.18 ”获取 上 传 文件 的 信息 


3.9.4 上 传 文件 函数 


Imove_uploaded_file0) 函 数 是 PHP 中 专门 用 来 上 传 文件 的 。 其 语法 格式 如 下 : 
bool move_uploaded file ( string filename, string destination ) 


功能 : 上 传 文件 到 指定 的 位 置 。 如 果 成 功 返 回 True， 否 则 返回 False。 

回 参数 说 明 : 有 两 个 参数 ， 分 别 介 绍 如 下 。 
人 “fename: 上 传 文件 的 临时 文件 名 ， 即 $_FILES['tmp_name'] 。 | 
仿 “destination: 上 传 后 保存 的 新 的 路 径 和 名 称 。 | 
通过 3.6.3 节 和 3.7.3 节 的 介绍 ， 细 心 的 读者 可 能 已 经 发 现 ， 真 正 的 上 传 文件 已 经 被 | 
保存 到 临时 目录 里 ，move_uploaded_file0) 函 数 的 作用 只 是 将 文件 换个 地 方 保存 ， 这 是 出 于 | 
安全 的 考虑 。 | 
此 外 ， 还 可 以 通过 is_uploaded_file0 函 数 判断 文件 是 否 为 POST 上 传 。 其 语法 格式 如 下 : 

bool is_uploaded file ( string filename ) 


3.9.5 多 文件 上 传 


多 文件 上 传 和 单 文件 上 传 没什么 区 别 ， 只 要 将 表单 中 的 文件 域 用 数组 命名 即 可 。 代 码 如 下 : 
01 <form action="2.php" method="post" enctype="multipart/form-data"> | 


02 <input name="userfile[]" type="file" /><br /> 
03 <input name="userfile[]" type="file" /><br /> 
i 

05 <input type="submit" value=" 上 传 " /> 

06 </form> 


89% 人 


| 说明; 


& Op 并 诅 模 决 精 角 


上 传 的 多 个 文件 可 以 通过 for 循环 语句 进行 获取 。 其 方法 如 下 : 
01 $filename=$ FILES[userfile][mname']: 


02 for($i=0; $i< count(Sfilename): $i++){ // 通 过 循环 将 多 个 文件 全 部 上 传 

03 move_uploaded file($tmpname[$i],$file path.$filename[$i]); 

04 // 添 加 数据 库 

05 Sinsertsql = "insert into tb_upfile (filename,filepath,filetype,upauthor,chkdownnum,ispub) 


06 values(".trim($filename[$1]).",".$file path.$filename[$1].","".$filetype[$1].", 
07 ™.$_SESSION['name'].",".trim($chkdownnum).",".$tmppub[$1].)'; 
08 S$conne->uidRst($insertsq]); 


上 
10 ”echo '<div style=" padding-top: 30px:"> 文 件 上 传 成 功 
11 <font color="#FF0000">'.$Schkdownnum.'</font>&nbsp; 对 文件 进行 下 载 。</div>'; 


本 模块 中 并 没有 加 入 多 文件 上 传 功能 ， 因 为 多 文件 上 传 存在 一 些 兽 端 ， 如 在 上 传 多 个 


文件 的 时 候 ， 多 个 文件 会 获取 相同 的 下 载 码 ; 在 上 传 多 个 文件 的 时 候 ， 判 断 多 个 文件 上 伟 
”是 否 有 重 名 比较 麻烦 ， 一 旦 上 传 失败 ， 会 导致 所 有 文件 都 上 传 失败 ， 必 须 重新 上 传 。 这 时 
可 以 在 上 传 页 面 中 增加 JavaScript 脚本 ， 在 提交 之 前 判断 是 否 存在 同名 文件 。 如 果 不 加 入 
”这 个 判断 过 程 ， 在 其 后 删除 多 个 同名 文件 的 时 候 就 会 导致 出 错 。 有 兴趣 的 读者 可 以 自行 研 
| 芝 = 


3.9.6 文件 下 载 


在 一 个 完整 的 网 站 中 ， 拥 有 文件 下 载 功能 能 够 给 网 站 带 来 更 多 的 访问 者 ， 提 高 网 站 的 


| 访问 量 。 实 现下 载 最 常用 的 方式 有 通过 链接 下 载 和 通过 header() 函 数 下 载 两 种 。 


(1) 通过 链接 下 载 

通过 链接 下 载 ， 就 是 将 文件 的 相对 路 径 或 绝对 路 径直 接 作为 超 链 接 来 下 载 。 如 果 是 压 
缩 文 件 ， 那 么 直接 单 击 超 链 接 就 可 以 下 载 ， 对 于 文本 文件 ， 单 击 鼠 标 右键 ， 在 弹出 的 快捷 
菜单 中 选择 “目标 另存 为 ”命令 ， 就 可 以 将 该 文件 下 载 下 来 。 所 有 格式 的 文件 ， 都 可 以 通 
过 这 种 方法 下 载 ， 运 行 结果 如 图 3.19 所 示 。 


您 想 打 开 或 保存 此 文件 四 ? 


和 名称，IE8.0 安 美 包 rar 


类 型 ，WinRAR 厅 绽 文件 20 字 五 
来 源 ，127001 


记 革 开 锯 保存 GS) | [了 沽 ] 


a sai 


图 3.19 通过 链接 下 载 
(2) 通过 header0 函 数 下 载 
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渤 
header0) 函 数 属于 HTTP 函数 ,其 作用 是 发 送 一 个 原始 的 HTTP 标 头 .其 语法 格式 如 下 :| 
void header ( string string [, bool replace [, int http_response code]] ) | 


参数 说 明 如 下 。 ee 


号 加 string:， 发 送 的 标 头 。 对 | 
旦 加 replace: 如 果 一 次 发 送 多 个 标 头 , 指明 相似 的 标 头 是 蔡 换 还 是 添加 。 如 果 是 False， 
则 强制 发 送 多 个 同类 型 标 头 ， 默 认 是 True， 即 替换 。 | 

号 加 http_response_code: 强制 HTTP 响应 为 指定 值 。 

HTTP 标 头 有 很 多 ， 这 里 只 介绍 下 载 ， 其 他 参见 相关 资料 。 下载 HTTP 标 头 的 代 | 
码 如 下 : | 

header('Content-Disposition: attachment; filename="filename"): 

上 面 的 代码 中 唯一 需要 改动 的 就 是 flename， 将 flename 蔡 换 为 要 下 载 的 文件 就 可 以 

完成 下 载 。 
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网 页 计数 器 模块 
( 铝 。 自学 视频 、 源 程序 ， 配套 资源 mn4\) 


在 浏览 网 页 时 经 常 可 以 看 到 网 站 的 首页 带 有 一 个 制作 精美 的 计数 器 ， 用 来 
显示 该 网 站 的 访问 量 。 它 不 仅 可 以 美化 网 站 的 页 面 ， 还 能 吸引 更 多 浏览 者 的 注 
意 力 ， 提 高 网 站 的 访问 量 。 网 页 计数 器 的 形式 多 种 多 样 ， 有 直接 以 数字 计数 存 
储 于 文本 或 数据 库 中 的 ， 也 有 以 图 形 方式 计数 存储 于 文档 或 数据 库 中 的 …… 通 
过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


a 


phl 


lad 


简单 数字 计数 器 

图 形 数 字 计 数 器 

利用 SESSION 实现 无 刷新 计数 器 
GD2 图 形 计 数 颂 

数据 库 数 字 计 数 器 

Cookie 计数 器 
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4.1 网 页 计数 器 模块 概述 


4.1.1 模块 概述 全 二 


网 页 计数 器 对 于 网 站 管理 者 来 说 是 一 个 非常 值得 关注 的 部 分 ， 它 记录 了 网 站 被 访问 的 1 yote 
次 数 ， 客 观 地 反映 了 网 站 受 欢迎 的 程度 。 如 果 网 站 的 点 击 率 很 高 ， 访 问 者 也 会 认为 网 站 值 | 
得 关注 ， 因 而 耐心 驻足 ， 进 而 增加 网 站 的 访问 量 ， 提 高 网 站 的 知名 度 。 | 


4.1.2 业务 流程 图 


网 页 计数 器 模块 主要 从 计数 器 功能 实现 的 方法 着 手 ， 详 细 地 介绍 了 简单 数字 计数 器 、 
图 形 数字 计数 器 、GD2 图 形 数字 计数 器 、 数 据 库 数字 计数 器 的 实现 方法 。 
网 页 计数 器 的 操作 流程 如 图 4.1 所 示 。 


判断 登录 标 


记 是 理 为 空 


为 登录 
标记 赋值 


网 站 访问 量 


将 数据 添加 
到 数据 库 中 
图 4.1 网 页 计数 器 的 操作 流程 

4.2 简单 数字 计数 器 


数字 计数 器 是 最 简单 的 一 种 ， 它 将 数据 存储 在 文本 文件 中 。 其 思路 是 首先 判断 文本 文 | 
件 是 否 存在 ， 如 不 存在 则 打开 失败 ， 如 打开 成 功 则 继续 执行 并 读 取 文件 中 的 数据 ， 将 计数 | 
器 的 值 加 1; 然后 以 写 的 方式 重新 打开 文件 ， 把 新 的 统计 数据 写 入 到 文件 中 ， 关 闭 文件 ; 
最 后 重新 打开 文件 ， 读 取 并 输出 文件 中 的 数据 。 

简单 数字 计数 器 的 操作 流程 如 图 4.2 所 示 。 


Er 四 


a tp 并 诅 模 决 精 角 


判断 文件 读 取 文 
是 否 存在 


以 写 入 的 
方式 打开 
文本 文件 


中 数据 ， 并 输出 文本 文件 中 
图 4.2 简单 数字 计数 器 的 操作 流程 
| 按 上 述 原理 设计 的 计数 器 ， 当 页 面 刷 新 时 其 值 就 会 增加 。 因 此 ， 要 想 控制 页 面 刷新 时 
| 计数 器 的 值 不 增加 ， 最 好 是 应 用 SESSION 变量 防止 重复 计数 。 
| 当 应 用 SESSION 防止 重复 计数 后 ， 无 论 如 何 刷 新 页 面 计数 器 的 值 都 不 会 增加 。 只 有 
”在 重新 打开 页 面 时 ,计数 器 的 值 才 会 继续 增加 。 简单 数字 计 数 器 的 运行 效果 如 图 4.3 所 示 。 


公司 理念 : 以 高 新 技术 为 依托 ， 战略 性 地 开发 具有 巨大 市 场 潜力 的 高 价值 的 产品 - 


公司 远景 : 成 为 镍 有 核心 技术 和 核心 产品 的 高 科技 公司 ， 在 某 些 领 域 具有 领先 的 市 场地 位 - 
核心 价值 观 : 永 葆 创业 激情 、 每 一 天 禾 在 进步 、 容 忍 失 败 ， 鼓 局 创新 、 充 分 信任 、 乎 等 交流 - 
网 站 当前 访问 量 : 100 
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图 4.3 简单 数字 计数 器 


下 面 对 简 单数 字 计 数 器 进行 详细 分 析 。 
| 首先 通过 session_start0) 函 数 初 始 化 一 个 SESSION 变量 ， 通 过 该 变量 来 控制 页 面 刷新 
”对 计数 器 的 影响 。 
| 然后 ， 通 过 站 语句 判断 SESSION 变量 的 值 是 否 为 空 ， 如 果 为 空 ， 则 打开 指定 的 文本 
| 文件， 否则 将 不 执行 。 
| 如 果 SESSION 变量 的 值 为 空 , 则 调用 fopen0 函 数 以 只 读 的 方式 打开 指定 的 文本 文件 ， 
| 如 果 文 件 不 存在 则 输出 打开 文件 失败 ， 否 则 将 继续 执行 ， 读 取 文 件 中 数据 。 这 里 将 指定 的 
| 文本 文件 countertxt 存储 在 根 目录 下 。 
| 如 果 文 本 文件 存在 ， 则 继续 执行 ， 将 调用 fgets0 函 数 读 取 文本 文件 中 的 数据 ， 并 通过 
”felose() 函 数 关闭 文件 。 
| 此 时 将 读 取 到 的 数据 $counter 加 1， 即 将 计数 器 的 值 加 1。 通 过 fopen() 函 数 重新 打开 
| 指定 的 文本 文件 (注意 ， 是 以 写 入 的 方式 打开 )， 并 调用 futs() 函 数 将 新 统计 的 数据 写 入 
”到 文本 文件 中 ， 关 闭 文件 。 
| 最 后 为 SESSION 变量 赋值 ， 将 SESSION 变量 赋值 为 1， 用 于 控制 因 网 页 刷新 而 导致 
”计数 器 的 值 无 限 增加 。 

简单 数字 计数 器 的 文件 存储 于 mr\4\01\index.php 文件 中 ， 其 关键 代码 如 下 。 


J wi 
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01 <?php | 
02 /使 用 文本 文件 存储 数据 | 
03 这 lisset($_ SESSION[temp])){ // 判 断 $_SESSION[temp]=="" 的 值 是 否 为 空 ， 其 中 的 temp ， 


04 if(($fp=fopen("counter.txt","r"))=——false){ | 贪 凡 
05 echo "打开 文件 失败 1"; | 一 一 
06 Yelse{ 

07 S$counter=fgets($fp,1024); // 读 取 文 件 中 数据 Note 
08 felose($fp): 1/ 关闭 文本 文件 | 

09 $countertt+; /1/ 计 数 器 的 值 加 1 | 

10 S$fp=fopen("counter.txt","w"); // 以 写 的 方式 打开 文本 文件 | 

11 fputs($fp,$counter); // 将 新 的 统计 数据 加 1 | 

12 felose($fp): | 

13 1/ 关闭 文件 | 

14 $_SESSION['temp']=1; /计数 器 的 值 增加 后 ， 为 $_ SESSION[temp] 赋 值 1 | 

15 


) 
16 /从 文本 文件 中 读 取 统计 数据 
17 if(($fp=fopen("counter.txt","r"))==false){ 


18 echo "打开 文件 失败 1"; 
19 }else{ 

20 $counter=fgets($fp,1024); 
21 felose($fp); 

22 

23 ?> 


(1) fopen() 函 数 
该 函数 用 于 打开 某 文 件 ， 并 返回 该 文件 的 标识 指针 。 其 语法 格式 如 下 : 


resource fopen(string filename, string mode [, int use_include_path [, resource context]]) 
其 中 参数 说 明 如 表 4.1 所 示 。 
表 4.1 fopen() 函 数 的 参数 说 明 


filename 必要 参数 。 用 于 指定 要 打开 文件 的 本 地 或 远程 地 址 
mode | 必要 参数 。 用 于 指定 打开 文件 的 模式 
可 选 参数 。 如 果 将 该 参数 设置 为 hue，PHP 会 尝试 按照 nclude_path 标 准 包含 路 径 中 
的 每 个 指向 去 打开 文件 | 
context 可 选 参数 。 设 置 提高 文件 流 性 能 的 一 些 选项 | 
在 fopen() 函 数 的 mode 参数 中 ， 其 指定 的 打开 模式 很 多 ， 其 中 常用 的 有 如 下 几 种 。 | 
回 :只 读 方式 打开 ， 将 文件 指针 指向 文件 头 。 | 
回 +': 读 写 方式 打开 ， 将 文件 指针 指向 文件 头 。 | 
回 ”'w': 写 入 方式 打开 ， 将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文件 不 存 | 
在 ， 则 尝试 创建 。 
回 'w+': 读 写 方式 打开 ， 将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文 件 不 
存在 ， 则 尝试 创建 


use_include_path 


o 
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到 .pp 并 晶 搞 决 精 角 


(2) fgets0 函 数 
该 函数 用 于 获取 文件 指针 所 在 行 指定 长 度 的 内 容 。 其 语法 格式 如 下 : 


4 | string fgets(int handle [,int length]) 
全 ss 数 说 明 如 下 : 
回 handle: 必要 参数 ， 指 调用 fopen0) 或 fsockopen0) 函 数 成 功 打 开 文 件 或 连接 上 服务 


器 后 所 返回 的 文件 标识 。 
回 length: 可 选 参数 , 用 于 指定 读 取 一 行 最 多 的 字 节 数 ， 并 返回 length-1 字 节 的 字符 
串 。 如 果 省 略 该 参数 ， 则 length 的 长 度 默认 为 1024 字 节 。 
(3) fputs0 函 数 
| fwrite0) 函 数 的 别名 , 用 于 将 字符 串 写 入 到 指定 的 文件 中 , 并 可 以 规定 写 入 字 节 的 大 小 。 
| 其 语法 格式 如 下 : 
| int fwrite( resource handle, string string [, int length]) 


其 中 参数 说 明 如 表 4.2 所 示 。 

表 4.2 fputs() 函 数 的 参数 说 明 
参数 说 明 
handle 
strin 
length 


必要 参数 。 文 件 标识 指针 
必要 参数 。 要 写 入 某 文件 的 字符 串 
可 省 参数 。 指 写 入 文件 的 长 度 ， 如 果 省 略 该 参数 则 将 指定 字符 串 的 所 有 内 容 写 入 到 文件 中 


至 此 ， 简 单数 字 计 数 器 的 计数 部 分 设计 完成 ， 接 下 来 要 做 的 就 是 在 网 页 中 重新 输出 文 


01 <?php 

02 /从 文本 文件 中 读 取 统计 数据 

03 if(($fp=fopen("counter.txt","r"))==false) { // 打 开 指 定 文件 
04 echo "打开 文件 失败 1"; 

05 }else{ 

06 Scounter=fgets($fp,1024):; // 读 取 文件 内 容 
07 fclose($fp): 

08 echo "数字 计数 器 : " .Scounter ; // 输 出 访问 次 数 
09 } 

i> 


4.3 图形 数字 计数 器 


| 所 谓 图 形 数字 计数 器 ， 就 是 将 计数 器 中 的 数据 以 数字 图 形 的 形式 显示 出 来 。 其 设计 原 
理 与 简单 数字 计数 器 类 似 ， 唯 一 区 别 的 地 方 就 是 在 对 数据 的 输出 过 程 中 ， 数 字 计数 器 直接 
| 将 数据 进行 输出 ， 而 图 形 数 字 计 数 器 则 将 数据 转化 成 数字 图 形 的 形式 进行 输出 。 其 运行 效 
| 果 如 图 4.4 所 示 。 


康 。96 。 
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明日 图 匠 绾 理 系统 > .3 得 末 和 


您 是 第 MOHOHOHCHO 访 问 本 系统 的 人 ! 简单 查询 高 仍 查 询 分 组 统计 退出 系统 反 
吉林 省 明日 科技 有 限 公司 是 一 家 以 计算 机 软件 技术 为 核心 的 高 科技 型 企业 ， 公司 外 建 于 2000 年 12 月 ， 是 专业 的 应 | 全 


用 软件 开发 商 和 服务 提供 商 。 多 年 来 # 


图 4.4 图 形 数字 计数 器 
计数 器 功能 的 实现 方法 已 经 在 简单 数字 计数 器 中 进行 了 详细 的 讲解 ， 这 里 不 再 次 述 。 
下 面 主要 讲解 如 何 将 文本 文件 中 的 数据 以 图 形 的 形式 进行 输出 
图 形 数字 计数 器 存储 于 mr\4\02\ 文 件 夹 下 ， 其 实现 关键 步骤 如 下 。 
首先 ， 读 取 文 本 文件 中 的 数据 ， 将 其 写 入 到 一 个 变量 中 。 
01 <?php session start(); 


| 
件 开 发 、 数 字 化 出 版 物 开发 制作 、 计 算 机 网 络 系 统 综合 
Note 
i | 
| 
| 


02 ifl!isset($_SESSION['temp']){ // 判 断 $_SESSION[temp]=="" 的 值 是 否 为 空 , 其 中 的 
temp 为 自 定义 变量 | 

03 if(($fp=fopen("counter.txt","r"))==false){ 

04 echo "打开 文件 失败 !"; 

05 }else{ 

06 S$counter=fgets($fp,1024); // 读 取 文件 中 数据 

07 felose($fp): // 关 闭 文本 文件 

08 $counter++; /计数 器 的 值 加 1 

09 Sfp=fopen("counter.txt","w"); /以 写 的 方式 打开 文本 文件 

10 fputs($fp,S$counter); // 将 新 的 统计 数据 加 1 

11 fclose($fp); 

12 } // 关 闭 文件 

13 $_SESSION['temp']=1; /为 $_ SESSION[temp] 赋 一 个 值 

14 } 

15 ?> 


然后 ， 将 读 取 的 数据 以 图 形 的 形式 进行 输出 。 这 里 设计 的 是 一 个 6 位 数字 的 图 形 计数 | 
器 ， 将 0~9 的 数字 图 形 存储 在 根 目录 下 的 images 文件 夹 中 。 步 又 如 下 : | 
(1) 通过 strlen0 函 数 获取 数据 的 长 度 。 | 
(2) 输出 6 位 数字 的 图 形 计数 器 中 的 空 值 ， 即 处 理 6 位 数字 中 补 位 值 0 的 输出 。 首 | 
先 应 用 str_repeatO 函 数 获取 6 位 数字 中 补 位 值 0 的 个 数 ; 然后 应 用 for 语句 , 根据 补 位 值 0 
的 个 数 〔Sstr)， 循 环 输出 代表 0 的 图 形 。 | 
(3) 将 文本 文件 中 存储 的 数据 以 图 形 的 形式 输出 。 应 用 for 循环 语句 ， 根 据 stlen0 | 
函数 获取 变量 $counter 的 长 度 ， 循 环 输出 指定 的 图 形 ， 应 用 switch 语句 为 数据 指定 不 同 的 
图 形 。 | 


01 Slen=strlen($counter); /获取 字符 串 的 长 度 

02 Sstr=str_repeat("0",6-$len); // 获 取 6-$len 个 数字 0 

03 for($i=0;$i<strlen($str):$i++){ // 获 取 变量 $str 的 字符 串 长 度 
04 Sresult=$str[$i]: 

05 Sresult='<img src=images/0.gif>'; 

06 echo $result: 1/ 循环 输出 $result 的 结果 


二 二 让 四 


tp 各 蜡 扫 志 糙 名 


07 } 

08 for($i=0:$i<strlen(Scounter):$i++){ // 获 取 字 符 串 的 长 度 

09 Sresult=$counter[$1]; // 获 取 数 据 

10 switch(Sresult) { // 为 不 同 的 数据 指定 不 同 的 图 形 
11 // 如 果 值 为 "0"， 则 输出 0.gif 图 片 

15 case "0"; Sret[$i]="0.gif’: break: 

13 case "1"; $ret[$1]="l1.gif’; break: 

14 case "2"; $ret[$1]="4.gif’; break; 

15 case "3"; $ret[$1]="3.gif’; break; 

16 case "4"; $ret[$1]="4.gif’; break:; 

17 case "5"; $ret[$1]="S.gif’; break; 

18 case "6"; $ret[$1]="6.gif’; break; 

19 case "7"; $ret[$1]="7.gif’; break; 

20 case "8"; $ret[$i]="4.gif' break: 

21 case "9"; $ret[$1]="9.gif’; break:; 

22 } 

23 echo "<img src=images/".Sret[Si].".>"; 1/ 输出 文本 文件 中 存储 的 数据 
24 

95 

26 ?> 


说 明 : 


| 行 输出 ， 如 图 4.5 所 示 。 


述 ， 直 接 讲解 如 何 将 读 取 的 数据 以 GD2 图 形 的 形式 进行 输出 。GD2 图 形 计数 器 存储 于 


(1) strlen0) 函 数 : 获取 指定 字符 串 的 长 度 。 
(2 ) str repeat() 函 数 : 将 指定 的 字符 串 重复 输出 。 


4.4 GD2 图 形 计 数 器 


采用 GD2 图 形 计数 器 ， 可 以 将 计数 器 中 存储 的 数据 以 GD2 函数 绘制 的 图 形 的 形式 进 


全 业 精 神 : 博 节 、 创 新 、 求 实 、 筷 行 

公司 理念 : 以 高 新 技术 为 依托 ， 战略 性 地 开发 具有 巨大 市 场 潜力 的 高 价值 的 产品 - 

公司 远景 : 成 为 镍 有 核心 技术 和 棱 必 产品 的 高 科技 公司 ， 在 某 些 领 域 具 有 领先 的 市 场地 位 。 
核心 价值 观 : 永 葆 创业 激情 、 每 一 天 帮 在 进步 、 容 忍 失败 ， 鼓 励 创新 、 充 分 信任 、 平 等 交流 - 


风 站 部 访问 者 :66 


日 Copyhights reserved 2008 吉林 省 明日 科技 有 限 公司 
客户 服务 邮箱 : mingrisoftBningrisoft con 


图 4.5 GD2 图 形 计数 器 
计数 器 中 的 数据 仍然 存储 在 文本 文件 中 ， 至 于 如 何 存 储 和 读 取 其 中 的 数据 这 


| mr4\03 文件 夹 下 ， 其 关键 实现 步骤 如 下 。 
(1) 创建 一 个 画布 ， 并 指定 画布 背景 的 颜色 和 字体 的 颜色 。 首 先 通过 imagecreate() 


| 函数 创建 画布 ， 然 后 调用 imagecolorallocate() 函 数 来 分 别 定义 背景 和 字体 的 颜色 
定义 字体 的 颜色 时 用 到 了 rand0 函 数 ， 通 过 其 在 指定 的 范围 内 随机 选择 颜色 )。 


J 
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有 不 再 资 


(其 中 在 


$9 
(2) 应 用 iconv0 函 数 对 指定 的 中 文字 符 进 行 转 换 ， 并 应 用 imagettfiext0 函 数 输出 中 | 
文字 符 串 。 
(3) 应 用 imagestring0) 函 数 输 出 计数 器 中 存储 的 数据 ， 并 应 用 imagepng0) 函 数 以 PNG | 
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格式 将 图 像 输出 到 浏览 器 或 文件 中 ， 通 过 imagedestroy0 函 数 销毁 图 像 。 | 
01 <?php | 
02 if(($fp=fopen("counter.txt","r"))—false){ 
03 echo "打开 文件 失败 !"; | 
04 jelse{ 
05 $counter=fgets($fp,1024); 
06 felose($fp): 
07 // 通 过 GD2 函数 创建 画布 
08 Sim=imagecreate(240,24): 
09 $gray=imagecolorallocate($im,255,255,255); // 定 义 背 景 | 
10 $color =imagecolorallocate($im,rand(0,255),rand(0,255),rand(0,255)); /定义 字体 颜色 | 
中 // 输 出 中 文字 符 | 
12 $text="couter:"; // 对 指定 的 中 文字 符 串 进行 转换 
13 // 输 出 网 站 的 访问 次 数 
14 imagestring($im,5,160,5, $text. $counter, $color): 
15 imagepng($im); 
16 imagedestroy($inm): /以 PNG 格式 将 图 像 输出 到 浏览 器 
17 
18 ?> 


(1) imagecreate() 函 数 
该 函数 用 于 创建 一 个 新 的 图 像 ， 返 回 一 个 图 像 标 识 符 。 其 语法 格式 如 下 : 
resource imagecreate ( int x_size, int y_size ) 
参数 x_size、y_size 为 图 像 的 尺寸 ， 单 位 为 像素 (pixel)。 
(2) imagecolorallocate() 函 数 | 
该 函数 将 返回 一 个 标识 符 ， 代 表 了 由 给 定 的 RGB 成 分 组 成 的 颜色 。 其 语法 格式 如 下 : 
int imagecolorallocate ( resource image, int red, int green, int blue) ! 
参数 说 明 如 下 。 
image: imagecreatetruecolor() 函 数 的 返回 值 。 | 
red、green 和 blue: 分 别 是 所 需要 颜色 的 红 、 绿 、 蓝 成 分 ， 其 值 是 0 一 255 的 对 玫 
或 者 十 六 进 制 的 0x00 一 0xFF。 | 
imagecolorallocate() 必 须 被 调用 以 创建 每 一 种 用 在 image 所 代表 的 图 像 中 的 颜色 。 第 一 | 
次 对 imagecolorallocate() 函 数 的 调 月 有 会 填充 背景 色 。 | 
(3) rand0 函 数 
该 函数 用 于 产生 一 个 随机 数 ， 并 返回 随机 数 的 值 。 其 语法 格式 如 下 : 
int rand ( int min, int max) 


如 果 没 有 提供 可 选 参数 min 和 max， 则 返回 0~RAND_MAX 之 间 的 伪 随 机 数 ， 如 果 | 


。99 。 L_ 


.App 内 晶 搞 半 精 角 


SS 


提供 可 选 参数 ， 则 返回 可 选 参数 范围 之 内 的 随机 数 。 


图 


(4) iconv () 函 数 


该 函数 用 于 将 一 种 已 知 的 字符 集 文件 转换 成 另 一 种 已 知 的 字符 集 文件 。 其 语法 格式 如 下 : 


iconv ( string in_charset, string out charset, string str) 


(5) imagettftext () 函 数 


该 函数 用 于 以 TTF (TrmeType Fonts) 字体 向 图 像 中 绘制 字符 串 。 其 语法 格式 如 下 : 
alray imagettftext ( resource image, float size, float angle, int x, int y, int color string fontfile, string text ) 


其 中 参数 说 明 如 表 4.4 所 示 。 
表 4.4 imagettftext() 函 数 的 参数 说 明 


参数 说 有 明 
image 图 像 资源 
size 字体 大 小 。 根 据 GD 版 本 不 同 ， 应 该 以 像素 大 小 GD1) 或 点 大 小 GD2) 指定 


下 到 上 的 文字 
文字 的 x 坐标 。 设 定 第 一 个 字符 的 基本 点 


(6) imagestring() 函 数 
利用 该 函数 可 实现 水 平地 绘制 一 行 字符 串 。 其 语法 格式 如 下 : 


bool imagestring ( int im, int font, int x, int y, string str, int col ) 


字体 的 角度 ， 顺 时 针 计算 ，0 度 为 水 平 ， 也 就 是 3 点 钟 的 方向 〈 由 左 到 右 ) ，90 度 则 为 由 


| y 文字 的 y 坐 标 。 设 定 字体 基线 的 位 置 ， 不 是 字符 的 最 底 端 

color 文字 的 颜色 

fontfile 字体 的 文件 名 称 ， 也 可 以 是 远 端 的 文件 
text 字符 串 内 容 


imagestring() 函 数 用 col 颜色 将 字符 串 str 绘制 到 im 所 代表 的 图 像 的 x,y 坐标 处 (这 是 


字符 串 左 上 角 坐 标 ， 整 幅 图 像 的 左上 角 为 0.0)。 如 果 font 是 1、2、3、4 或 5， 则 使 用 


字体 。 


内 置 


至 此 ，GD2 图 形 计数 器 讲解 完毕 。 上 述 介绍 的 这 3 种 计数 器 都 是 从 输出 的 方式 上 进行 


区 别 的 ， 其 统计 的 方式 是 完全 相同 的 ， 下 面 将 要 继续 介绍 几 种 完全 不 同 的 计数 器 。 


4.5 ”数据库 数字 计数 器 


顾名思义 数据 库 数 字 计 数 器 ， 就 是 将 统计 的 数据 存储 到 数据 库 中 。 相 对 于 其 他 几 种 计 


“1005 


数 器 而 言 ， 数 据 库 数字 计数 器 的 功能 是 最 为 强大 的 ， 不 但 可 以 统计 出 一 个 数值 ， 还 可 以 记 
录 更 多 的 信息 (如 卫 地 址 、 访 问 时 间 等 )， 这 些 都 可 以 存储 在 数据 库 中 。 
4.6 所 示 。 


其 运行 效果 如 
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用 软件 开发 商 和 服务 可 洪 商 。 多 年 未 始 给 致 力 于 行业 管理 软片 开发 、 淘 了 化 出 版 鞠 开 友 抽 作 、 计 算 机 网 络 系统 乏 辣 ! 其 
全 业 宗 号 : 为 企业 服务 ,打造 企业 智能 着 短 平台 , 改 备 企业 的 管理 与 到 作 过 程 ， 提高 全 业 效 京 ， 降低 管理 或 二， 增 一 一 
芒 必 价值 观 : 承 葆 创业 委 傅 、 每 一 天 者 在 运 步 、 容 妃 和 网， 牙 德 新 ， 元 分 信 位、 平等 交流 。 

合计 cp 甸 呈 童工 Note 


Copyiizhts reserved 2008 吉林 省 明日 科技 有 限 公司 
害 记 服务 邮箱 ?ningrisof laniseriso5t com 


图 4.6 数据 库 数字 计数 器 


既然 是 数据 库 数字 计数 器 ， 就 需要 数据 库 的 支持 。 首 先 要 创建 一 个 数据 库 ， 并 在 其 中 
创建 一 个 数据 表 用 于 存储 数据 。 创 建 的 数据 表 如 图 4.7 所 示 。 


aulb_increment | 


Courts varchar(so) gu2312_chinese_ ol 


aa .varchars0) gb2312_chinese_cl | 
ala? varcharst) gt2312_chinese_ol ! 
mvarchar(s0) ob2312_chinese df 


图 4.7 创建 的 数据 表 | 
然后 连接 数据 库 。 连 接 数据 库 文件 m4\04\conn\eonn.php 的 代码 如 下 : 


01 <?php 

02 S$id=mysql_connect('localhost','root’,'111"); 1/ 连接 数 据 库 服务 器 

03 mysql_select_db(db_counter,$id): // 连 接 db_counter | 

04 mysql_query("set names gb2312"); /指定 数据 库 中 字符 串 的 编码 格式 | 
为 gb2312 | 

05 ?> 


注意 : 
在 连接 数据 库 的 过 程 中 ， 要 注意 数据 库 服务 器 、 用 户 名 和 密码 的 使 用 ， 还 有 连接 的 数 
据 库 要 正确 ， 以 及 对 数据 库 中 字符 囊 编码 格式 的 转换 。 


最 后 实现 网 站 访问 信息 的 存储 ， 并 且 将 访问 量 和 Pp 访问 量 以 图 形 图 像 的 形式 进行 输出 。 
数据 库 数字 计数 器 存储 于 mr4\04\ 文 件 夹 下 ， 其 关键 实现 步骤 如 下 。 
(1) 初始 化 一 个 SESSION 变量 ， 与 数据 库 建 立 连 接 ， 获 取 当 前 时 间 和 客户 端的 人 P 
地 址 ， 并 判断 SESSION 变量 的 值 是 否 为 空 。 
01 <?php 


02 session_ start(); | 
03 include("conn/conn.php"); | 


04 ~ $datal=date("Y-m-d h-m:s"): // 获 取 当 前 访问 时 间 | 
05 $data2=date("Y-m-d"); // 获 取 当 前 访问 时 间 | 
06 $ip=getenv(REMOTE ADDR'): | 
07 if($_ SESSION['temp']—""){ // 判 断 $_SESSION[temp] 一 "" 的 值 是 否 为 空 , 其 中 的 | 
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emp 为 自 定义 变量 


| (2) 通过 数据 库存 储 网 站 被 访问 的 数据 ， 执 行 SELECT 查询 ， 判 断 当 前 获取 的 他 地 
， | 址 在 数据 库 中 是 否 已 经 存在 。 如 果 数 据 库 中 存在 该 了 地 址 ， 则 执行 UPDATE 更 新 语句 ， 

做 F_ | ”将 该 JP 地 址 对 应 的 数据 的 访问 量 加 1， 并 且 更 新 访问 的 时 间 ， 如 果 数 据 库 中 不 存在 该 卫 
_ 地 址 ， 则 执行 NSERT 添加 语句 ， 将 该 他 地 址 的 数据 添加 到 数据 库 中 。 
01 $query=mysql_ query("select * from tb_count04 where ip=$ip'); 
02 Sresult=mysql_num _ rows($query); 
| 03 if($result>0){ 
04 $query_l=mysql query("update tb count04 set counts=countst+1,datal='$datal',data2= 
| 9$data2' where ip="$ip""); 


Note 


05 jelse{ 

06 Squery="insert into tb_ count04(counts,datal,data2.ip)values (1','$datal','$data2','$ip)"; 

07 Sresult=mysql_query($query); 

08 y 

(3) 将 SESSION 变量 赋值 为 1， 防止 因 页 面 刷新 导致 重复 计数 。 

01 $_SESSION['temp']=1: /登录 以 后 ，$_SESSION[temp] 的 值 不 为 空 ， 给 
| $ SESSION[temp] 赋 一 个 值 1 
| 02 > 

‘03 9> 


至 此 ， 数 据 库 数 字 计 数 器 介绍 完毕 。 
4.6 网 站 访问 量 统计 分 析 


计数 器 中 的 有 些 功能 已 经 在 前 面 的 章节 中 讲解 过 了 ,例如 采用 数据 库 来 统计 网 站 的 访 
问 量 :通过 SESSION 来 防止 网 站 的 重复 计数 ; 应 用 图 形 数字 计数 器 来 输出 网 站 当前 的 访 
| 问 量 等 。 下 面 介 绍 如 何 对 网 站 访问 量 的 数据 进行 统计 分 析 。 这 里 主要 应 用 jpgraph 类 创建 
| 柱 形 图 ， 对 网 站 访问 量 的 统计 结果 进行 分 析 ， 如 图 4.8 所 示 。 

EF | 


查看 访问 量 
网 站 总 的 访问 量 ; 177 网 站 当月 的 访问 量 : 1 殉 站 当日 的 沪 问 里 : 
网 站 总 的 1p 访问 量 : 1 当月 rr 的 访问 量 ; 1 现 站 当日 于 的 访问 量 : 0 
zm = 国 辐 


好 20104 月 由 浪 5 国 相国 


2011-0 月 网 站 访问 是 走势 


图 4.8 对 网 站 访问 量 的 统计 结果 进行 分 析 
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4.6.1 ”通过 数据 库 记 录 网 站 访问 量 | 
有 关 通 过 数据 库 统计 网 站 访问 量 的 具体 方法 已 经 在 4.5 节 中 作 了 详细 讲解 ， 这 里 不 再 | 


资 述 。 输 出 数据 库 中 存储 的 网 站 访问 量 应 用 的 是 4.3 节 的 图 形 数字 计数 器 ， 其 关键 代码 如 下 : | 会 内 
01 <Iphp os 
02 // 以 图 形 的 形式 输出 数据 库 中 的 记录 数 Note 
03 $querys="select sum(counts) as ll from tb_count10 ";// 查 询 数 据 库 中 总 的 访问 量 | 
04 $results=mysql query($querys,$id); | 
05 Sfwl=mysql result($results,0,"11"): | 
06 echo "----------"; | 
07 /对 补 位 数字 0 的 处 理 | 
08 Slen=strlen($fw]): /获取 字符 串 的 长 度 | 
09 S$str=str_repeat("0",6-$len); // 获 取 6-Slen 个 数字 0 | 
10 for($i=0;$i<strlen($str):$i++){ // 获 取 变量 $str 的 字符 串 长 度 | 
11 Sresultes=$str[$i]: | 
12 $resultes='<img src=images/0.gif>"; 

13 echo $resultes:; 1/ 循环 输出 $result 的 结果 

14 } 

15 /对 数据 库 中 数据 的 处 理 

16 for($i=0:$i<strlen(SfwD:SitH{ /获取 字符 串 的 长 度 

Ww Sresult=$fwl[$1]; 

18 switch($result){ 

19 // 如 果 值 为 "0", 则 输出 0.gif 图 片 

20 case "0"; $ret[$1]="0.gif";break: 

2 case "1"; $ret[$i]="1.gif':break: 

22 case "2"; $ret[$1]="2.gif":break:; 

23 case "3"; $ret[$i]="3.gif':break: 

24 case "4"; $ret[$i]="4.gif':break: 

25 case "5"; $ret[$i]="S.gif':break: 

26 case "6"; $ret[$i]="6.gif':break: 

27 case "7"; $ret[$1]="7.gif";break; 

28 case "8"; $ret[$i]="8.gif':break: 

29 case "9"; $ret[$i]="9.gif':break: ! 
30 | 
31 echo "<img src=images/".$ret[$i].".>"; // 输 出 访问 次 数 | 
32 } | 
33 ?> | 


网 站 访问 量 的 输出 结果 如 图 4.9 所 示 。 


ee 网 站 访问 量 统计 分 析 | 


图 4.9 输出 网 站 访问 量 | 
4.6.2 ”统计 分 析 网 站 访问 量 


在 图 4.9 所 示 的 网 站 访问 量 输出 结果 中 ， 有 一 个 “网 站 访问 量 统 计 分 析 ” 超 链接 ， 它 
“403® 
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可 


| 链接 到 indexs.php 文件 中 ， 在 此 文件 中 完成 对 网 站 访问 量 的 统计 操作 ， 其 统计 的 内 容 如 
| 图 4.10 所 示 。 


A 
Note i 二 


R801P 访 间 量 : 1 当日 JP99 访 i 量 : 0 


;009491396) 049T0902。 贡 砍 下拉 天 公司 诛 权 所 有 


图 4.10 网 站 访问 量 统计 


具体 实现 步 又 如 下 。 
(1) 连接 数据 库 ， 调 用 指定 的 数据 库 连接 文件 。 


<?php include("conn/conn.php"); ?> 


| (2) 输出 网 站 总 的 访问 量 以 及 总 的 他 访问 量 。 应 用 SELECT 语句 和 sum0) 函 数 ， 从 
| 数据 表 中 读 取 counts 字段 总 的 数据 ， 获 取 网 站 总 的 访问 量 。 


01 <td width="240" height="25" align="center" valign="middle"> 


02 <?php 

03 $query_1="select sum(counts) as countes from tb_count10 "; 

05 Sresult_1=mysql query($query_1):; 

06 S$countes_1=mysql] result($result_1,0,'countes'); 

07 echo "<p class='STYLE1 人 > 网 站 总 的 访问 量 : $countes_1</p>"; 


| (3) 获取 网 站 总 的 他 访问 量 。 从 数据 表 中 读 取 数 据 ， 应 用 mysql_fetch_array() 函 数 获 
| 取 p 字段 对 应 的 了 P 地址， 并 将 获取 到 的 值 赋 给 一 个 数组 ， 通 过 array_unique() 函 数 去 除数 
| 组 中 重复 的 值 ， 最 终 获 取 到 网 站 总 的 卫 访问 量 。 


04 // 查 询 数据 库 中 总 的 访问 量 


01 // 查 询 数据 库 中 总 的 下 访问 量 

02 $query_2="select * from tb_count10 "; 

03 Sresult_ 2=mysql query($query_2): 

04 while($myrow=mysql fetch array($result 2)){ 

05 S$counts_ 2[]=$myrow['ip']; /将 获取 的 ip 值 赋 给 变量 

06 } 

07 Sresults 2=array_unique($counts 2); // 去 除数 组 中 重复 的 值 

08 $countes 2=count($results 2); 1/ 获取 数组 中 值 的 数量 ， 即 总 的 下 访问 量 
09 echo "<p class='STYLE1 人 > 网 站 总 的 他 访问 量 : $countes 2</p>"; 

10 ?></td> 


| (4) 获取 网 站 当月 的 访问 量 和 当月 IP 的 访问 量 。 使 用 的 方法 与 获取 网 站 总 的 访问 量 
和 总 的 卫 访问 量 的 方法 类 似 ， 区 别 在 于 SELECT 语句 中 增加 了 一 个 查询 条 件 
data2=".substr(date("Y-m-d").0.7)."， 即 查询 数据 库 中 符合 当前 月 份 的 数据 。 
| 01 <td width="240" align="center" valign="middle"> 

| 02 <?php 

03 // 查 询 数 据 库 中 当月 总 的 访问 量 
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04 $query_3="select sum(counts) as countes from tb count10 where data2=".substr | 
(date("Y-m-d"),0,7)."": | 

05 Sresult 3=mysql query($query_3): | 

06 S$countes 3=mysql result($result 3,0,'countes"); | 

07 echo "<p class='STYLE2> 网 站 当月 的 访问 量 : $countes 3</p>"; | 优 欠 

08 /查询 数据 库 中 当月 的 卫 访问 量 | ~ 

09 $query_4="select * 他 om tb_count10 where data2=".substr(date("Y-m-d"),0,7).""; 

10 Sresult_ 4=mysql query($query_4); a Note 

11 Scounts_4=array(); | 

12 while($myrow_4=mysql fetch array($result 4)){ | 

13 $counts 4[]=$myrow_ 4['ip']; // 将 获取 的 他 值 赋 给 变量 | 

14 } | 

15 S$results 4=array_ Unique($counts 4); /去 除数 组 中 重复 的 值 | 

16 S$countes 4=count($results 4); // 获 取 数 组 中 值 的 数量 ， 即 总 的 他 访问 量 | 

17 echo "<p class='STYLE2> 网 站 当月 IP 的 访问 量 : $countes_4</p>"; | 

18 2></td> | 


(5) 获取 网 站 当日 的 访问 量 和 当日 的 下 访问 量 。 其 方法 这 里 不 再 资 述 ， 指 定 的 查询 
条 件 是 datal=".date("Y-m-d")."。 程 序 关 键 代码 如 下 : | 


01 <td width="248" align="center" valign="middle"> 

02 <?php 

03 // 查 询 数 据 库 中 当日 总 的 访问 量 

04 $query_5="select sum(counts) as countes from tb_count10 where datal=".date ("Y-m-d").™" "; 

05 Sresult 5=mysql query($query_5):; 

06 $countes 5=mysql] result($result_5,0,'countes'); 

07 echo "<p class='STYLE3> 网 站 当日 的 访问 量 : $countes_5</p>"; 

08 /查询 数据 库 中 当日 的 正 访问 量 

09 $query_6="select * from tb_count10 where datal=".date("Y-m-d")."™"; 

10 Sresult_6=mysql query($query_6):; 

11 S$counts_6=array(); 

12 while($myrow_6=mysql fetch array($result_6)){ 

13 S$counts_6[]=$myrow_6['ip"]; /将 获取 的 他 值 赋 给 变量 

14 

15 ifis_array($counts 6)){ 

16 Sresults_6=array_unique($counts 6); // 去 除数 组 中 重复 的 值 | 
17 $countes_6=count(Sresults 6); /获取 数组 中 值 的 数量 ， 即 总 的 卫 访问 量 | 
18 echo "<p class='STYLE3 作 网 站 当日 人 P 的 访问 量 : $countes_6</p>"; | 
To | 
20 <d> ! 


4.6.3” 柱 形 图 分 析 网 站 月 访问 量 比重 


在 indexs.php 文件 中 ， 通 过 下 拉 列 表 框 选择 指定 的 月 份 ， 单 击 “ 提 交 ” 按 钮 ， 将 在 本 
页 以 柱 形 图 的 形式 输出 网 站 指定 月 份 的 访问 量 比重 。 | 
首先 ， 创 建 表单 ， 提 交 要 分 析 的 指定 月 份 。 其 关键 代码 如 下 : | 


01 <form name="forml" method="post" action="indexs.php"> | 
02 <select name="select"> 
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03 <2php 
04 $que="select distinct data2 from tb_count10": /查询 数据 库 中 月 份 的 数据 
05 $res=mysql query($que,$1id); 
06 while($myrow=mysql fetch array($res)){ 1/ 循环 输出 月 份 数据 
07 ?> 
08 <option value="<?php echo $myrow['data2"]:?>"><?php echo $myrow['data2']:;?> 
</option> 
09 <?php 
10 让 
11 ?> 
12 </select>&nbsp:&nbsp: 
13 <input type="submit" name="Submit" value=" 提 交 "> 


14 </form> 


然后 ， 在 本 页 中 获取 表单 提交 的 月 份 数 据 ， 以 表单 提交 的 数据 为 条 件 进行 查询 ， 将 从 


数据 库 中 读 取 的 数据 存储 到 指定 的 数组 中 ， 并 将 数组 重新 生成 新 字符 串 ， 将 新 字符 串 作 为 
” 超 链 接 的 参数 值 传递 到 statphp 文件 中 。 其 关键 代码 如 下 : 


01 <?php 

02 ifisset($_POST['select])){ // 判 断 提交 的 数据 是 否 为 真 
| 03 $query="select counts,datal from tb countl0 where data2=".$ POST['select]." order by 
datal "; // 定 义 SQL 语句 

04 $result=mysql_query($query); // 执 行 查询 语句 

05 Sresults=array(); // 定 义 数 组 

06 S$datas=array(); // 定 义 数 组 

07 while($myrow=mysql fetch array(Sresult)){ /循环 输出 查询 结果 

08 Sresults[]=$myrow['counts']; // 将 查询 结果 存储 到 数组 中 

09 Slmbs=implode(",", $results); /将 数组 重新 生成 一 个 字符 串 

10 $datas[]=substr($myrow["datal'].8.2): // 截 取 时 间 数 据 ， 并 存储 到 数组 中 

11 $das=implode(",",$datas): /将 数组 重新 生成 一 个 字符 串 

12 } 

13 > 


14 <table width="902" border="1"” cellpadding="1"” cellspacing="1"” bordercolor="#FFFFFF" 


| bgcolor="#D5D5D5"> 


J <tr> 

16 <td height="18" background="images/mysql_9.jpg" bgcolor="#FFFFFF" 
class="STYLE4"> 

7 <?php echo $_POST['select]:?> 月 网 站 访问 量 走势 图 

18 </td> 

19 </tr> 

20 <tr> 

po <td bgcolor="#FFFFFF"> 

2 <p><img src="stat.php?lmbs=<?php echo S$lmbs: ?>&das=<?php echo 
$das;?>"/></p> 

23 <p align="center" class="STYLE2"><?php echo $_ POST['select]:?> 月 网 站 访问 
量 走势 图 <p> 

24 </td> 

25 </tr> 

26 </table> 
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27 ”<?php | 
25 | 
29 ?> 


最 后 ,在 statphp 文件 中 ， 应 用 jpgraph 类 库 创 建 柱 形 图 ， 以 超 链接 传递 的 参数 值 作为 
数据 ， 生 成 网 站 月 访问 量 统计 图 。statphp 文件 的 代码 如 下 : 


01 <?php 

02 include ("src/ipgraph.php"); // 载 入 类 库 文件 

03 include ("src/jpgraph_bar.php"); 

04 $lmbs=$ GET['Imbs"]; // 获 取 超 链接 传递 的 参数 值 

05 $das=$ GET['das']; | 
06 $datay=explode(",",Slmbs); // 按 照 "," 分 隔 字 符 串 , $data 为 获取 的 数据 变量 | 
07 Sdatax=explode(",",$das): // 按 照 "," 分 隔 字 符 串 , $data 为 获取 的 数据 变量 | 
08 S$graph=new Graph(800,200,"auto"); /创建 图 像 | 
09 $egraph->img->SetMargin(60,20,30,50); // 设 置 图 像 边框 间距 | 
10 $egraph->SetScale("textlin"); /定义 坐标 刻度 类 型 | 
11 Sgraph->SetMarginColor("lightblue"); // 定 义 图 像 颜色 | 
12 S$graph->title->Set('counter"); // 定 义 标题 

13 $egraph->title->SetFont(FF_SIMSUN,FS BOLD):; // 设 置 标题 字体 


14 ”Sgraph->xaxis->SetfFont(FF_VERDANA,FS NORMAL,10); /设置 和 X 轴 的 字体 
15 ”S$graph->yaxis->SetFont(FF_VERDANA,FS NORMAL,10); /设置 立轴 的 字体 


16 S$graph->xaxis->SetTickLabels($datax); /设置 X 轴 输 出 的 数据 

17 S$graph->xaxis->SetLabelAngle(50); /设置 输出 文字 大 小 

18 S$bplot = new BarPlot($datay); // 实 例 化 图 像 创 建 类 

19 $bplot->SetWidth(0.6); // 设 置 柱 形 图 的 输出 大 小 | 

20 $bplot->SetFillGradient("navy","#FFFF00",GRAD_LEFT_REFLECTION); /设置 图 像 的 | 
类 型 和 填充 颜色 

21 $bplot->SetColor("white"): /设置 图 像 边框 颜色 

22 $egraph->Add($bplont): // 添 加 数据 

23 $egraph->Stroke(): // 生 成 图 像 

24 ?> 

4.7 技术 提炼 


网 页 计数 器 的 设计 方法 很 多 ， 应 用 不 同方 法 设计 的 计数 器 的 实效 性 也 不 尽 相同 。 下 面 
对 网 页 计数 器 开发 中 应 用 到 的 技术 进行 提炼 。 


4.7.1 防止 重复 计数 
在 开发 网 页 计数 器 的 程序 中 ， 一 个 最 为 关键 的 问题 就 是 如 何 防止 网 站 的 重复 计数 。 如 | 


果 一 个 计数 器 不 能 控制 重复 计数 ， 那 么 这 个 计数 器 将 没有 任何 意义 。 根 据 设计 计数 器 的 方 | 
法 不 同 ， 总 结 出 两 种 控制 重复 计数 的 方法 。 | 


1. 通过 SESSION 防止 重复 计数 | 
此 方法 可 以 应 用 到 任意 一 个 计数 器 中 实现 控制 重复 计数 的 功能 。 其 实现 的 原理 是 : 在 
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这 % 
当前 页 被 访问 时 ， 初 始 化 一 个 SESSION 变量 ， 并 赋 给 其 一 个 空 值 ， 然 后 判断 SESSION 变 
| 量 的 值 是 否 为 空 ， 如 果 为 空 ， 则 将 计数 器 的 值 加 1， 并 且 为 SESSION 变量 赋值 1。 
| 此 时 , 在 当前 页 中 , SESSION 变量 的 值 已 经 不 为 空 , 无 论 如 何 刷新 , 其 值 都 不 会 改变 ， 
做 A” 所 以 计数 器 的 值 也 不 会 增加 。 关 键 代码 如 下 


| 01 <?php 
02 /使 用 文本 存储 数据 
| 03 ”if(!isset($_SESSION['temp"])){ /判断 S_SESSION[temp]=="" 的 值 是 否 为 室 ， 其 中 的 temp 
| 为 自 定义 变量 
| 04 if(($fp=fopen("counter.txt","r"))==false){ 
05 echo "打开 文件 失败 1"; 
06 }else{ 
07 $counter=fgets($fp,1024); 1/ 读 取 文件 中 数据 
08 felose($fp); 1/ 关闭 文本 文件 
09 $counter++; /计数 器 的 值 加 1 
10 S$fp=fopen("counter.txt","w"):; // 以 写 的 方式 打开 文本 文件 
11 fputs($fp,ScounteD: /将 新 的 统计 数据 加 1 
12 felose($fp): 
13 } /关闭 文件 
14 $_SESSION['temp']=1; /计数 器 的 值 增 加 后 ， 为 $ SESSION[temp] 赋 值 1 
15 


16 /从 文本 文件 中 读 取 统计 数据 
i if(($fp=fopen("counter.txt","r"))=—false){ 


18 echo "打开 文件 失败 1"; 
19 }else{ 

20 $counter=fgets($fp,1024); 
2 fclose($fp); 

2 

23 ?> 


| 琶 . 
| SESSION 变量 防止 重复 计数 的 缺点 是 只 对 当前 页 面 有 效 ， 当 重新 打开 该 页 面 时 ， 计 数 
| 器 的 值 仍然 会 增加 。 


2. 通过 JP 防止 重复 计数 


| 通过 并 防止 重复 计数 是 最 准确 的 方法 ， 它 完全 是 通过 客户 端 机 器 的 他 来 防止 重复 计 
， 数 ， 不 会 有 任何 时 间 和 页 面 的 限制 ， 只 要 是 已 经 访问 过 的 卫 ， 计 数 器 的 值 就 不 会 增加 。 

”通过 IP 防止 重复 计数 必须 要 有 数据 库 的 支持 ， 服 务 器 会 将 每 个 访问 过 本 网 站 的 客户 
端的 IP 存储 到 数据 库 中 。 

。 当 客户 端的 机 器 访问 本 网 站 时 ， 服 务 器 将 首先 获取 该 客户 端的 卫 地 址 ， 并 且 与 数据 
， 库 中 存储 的 他 地 址 进行 比较 ， 如 果 数 据 库 中 已 经 存在 此 全 ， 则 计数 器 的 值 将 不 会 增加 ; 
”否则 将 计数 器 的 什 加 1， 并 且 将 该 全 地址 存储 到 数据 库 中 。 关 键 代 码 如 下 : 


01 <?php 
02 session start(); 
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03 include("conn/conn .php'); 


04 $datal=date("Y-m-dh:m:s"); // 获 取 当 前 访问 时 间 

05 $data2=date("Y-m-d"); // 获 取 当 前 访问 时 间 

06 Sip=getenv(REMOTE_ADDR'); | 

07 /使 用 数据 库存 储 数据 | 到 六 
08 $query=mysql query("select * from tb_count04 where ip="'Sip""); | i 
09 Sresult=mysql num rows($query); 

10 if(Sresult>0){ 
i echo "该 他 地 址 已 经 存在 "; | 

12 jelse{ 

13 $query="insert into tb_count04(counts,datal ,data2,ip)values('1','$datal','$data2','$ip)"; 

14 Sresult=mysql_query($query); 

15 00 

16 ?> 


通过 耳 不 但 可 以 防止 重复 计数 ， 而 且 可 以 统计 出 网 站 被 同一 卫 访问 的 次 数 。 
4.7.2 通过 array_unique() 函 数 获取 网 站 当月 IP 访问 量 


通过 array_unique() 函 数 去 除数 组 中 重复 的 值 ， 可 以 实现 统计 网 站 当月 IP 访问 量 。 其 | 
中 主要 用 到 了 array_unique0 函 数 和 count0 函 数 。 | 
(1) array_unique() 函 数 
该 函数 将 值 作为 字符 串 排序 , 然后 对 每 个 值 只 保留 第 一 个 键 名 , 忽略 后 面 所 有 的 键 名 。 
其 语法 格式 如 下 : 


array array_unique ( array array) 


(2) count0 函 数 
该 函数 用 于 统计 数组 中 元 素 值 的 个 数 。 其 语法 格式 如 下 : 


int count ( mixed array [, int mode]) 


参数 说 明 如 下 。 
array: 指定 的 数组 。 | 
mode: 可 选 参 数 ，COUNT_RECURSIVE (或 1)。 如 选中 此 参数 ， 本 函数 将 递归 ， 
地 对 数组 计数 。 这 对 于 计算 多 维 数组 的 所 有 单元 尤其 有 用 。 此 参数 的 默认 值 是 0。 | 
在 获取 网 站 当月 IP 访问 量 之 前 ， 先 来 看 一 下 存储 网 站 访问 量 数据 的 数据 表 的 结构 ， | 
如 图 4.11 所 示 。 


胃 服务 器 : localhost ， 昌 数据 库 : db_counter ， 国 表 :tb_count10 
| 同 浏 和 了 结构 “了 sol 万 搜索 也 项 入 别 导 出 _ 阐 mport 作 拨 作 [ 独 请 空 罗 开除 
有 关 型 束 理 性 性 Nul 至 认 。 蕊 外 操作 
厂 了 由。 Ito) 理 auto_inerement 图  X 加 
厂 counts varchar(50) gb2312_chinese_ci 理 国志 XX 国 
厂 datat varchar(50) gb2312 殖 萎 X 国 
厂 data2 varchar50) gb2312.¢ 理 克 X 图 
厂 pvarchars0) gb2312_chinese_i 百 国 ZX EE 


图 4.11 网 站 访问 量 统计 数据 表 
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在 该 数据 表 中 ， 包 括 5 个 字段 ， 分 别 是 ID; 网 站 访问 量 counts; 访问 时 间 datal ， 完 


， 整 的 时 间 年 月 日 ， 访 问 时 间 data2， 只 包括 年 和 月 ， 网 站 的 他 地 址 。 


要 获取 网 站 当月 人 P 访问 量 ,其 查询 的 条 件 必须 从 时 间 data2 着 手 ， 查 询 出 当前 时 间 符 


| 合 data2 的 数据 。 


<2php 
$query_4="select * from tb_ count10 where data2="'".substr(date("Y-m-d"),0,7)."""; 
Sresult 4=mysql query($query_4): 


| 技巧; 


应 用 substr0) 函 数 ， 获 取 当 前 日 期 中 年 和 月 的 值 ， 将 获取 的 值 与 数据 表 中 data2 字段 的 


| 值 进行 比较 ， 从 中 读 取出 符合 条 件 的 数据 。 


然后 ， 将 获取 到 的 卫 地 址 写 入 到 一 个 数组 中 。 


while($myrow_4=mysql fetch _array($result 4)){ 
S$counts_2[]=Smyrow['ip']; // 将 获取 的 ip 值 赋 给 变量 
} 


最 后 ， 应 用 array_unique() 函 数 去 除数 组 中 重复 的 值 ， 再 统计 出 数组 中 元 素 的 数量 ， 即 


| 总 的 耳 访问 量 。 


Sresults_4=array_unique($counts 4); /去 除数 组 中 重复 的 值 
$countes 4=count($results 4): // 获 取 数 组 中 值 的 数量 ， 即 总 的 他 访问 量 
?> 


4.7.3 将 数组 中 的 数据 写 入 到 字符 串 中 


将 数组 中 的 数据 写 入 到 字符 串 中 ， 主 要 应 用 的 是 current 0 函数 和 implode 0 函数 。 
(1) current0 函 数 
每 个 数组 中 都 有 一 个 内 部 指针 ， 初 始 指 向 插入 到 数组 中 的 第 一 个 单元 。 该 函数 返回 当 


| 前 被 内 部 指针 指向 的 数组 单元 的 值 ， 并 不 移动 指针 。 如 果 内 部 指针 指向 超出 了 数组 的 最 大 
”单元 ， 将 返回 false。 其 语法 格式 如 下 : 


mixed current ( array array) 
(2) implode() 函 数 
该 函数 用 于 将 数组 的 内 容 组 合成 一 个 字符 串 。 其 语法 格式 如 下 : 
string implode(string glue, array pieces) 
参数 glue 是 字符 之 间 的 分 隔 符 。 
这 里 实际 是 将 指定 月 份 每 天 的 数据 写 入 到 字符 串 中 ， 实 现 对 网 站 指定 月 份 的 访问 量 进 


行 分 析 。 要 完成 该 操作 ， 首 先 要 以 指定 的 月 份 为 查询 条 件 ， 查 询 出 指定 月 份 的 网 站 访问 量 
| 的 数据 。 


<?php 
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$query="select counts,datal from tb_count10 where data2="".$_ POST['select]." order by datal "; 
Sresult=mysql_query($query); 


然后 ， 通 过 while 语句 ， 循 环 读 取 其 中 的 数据 ， 并 且 应 用 currentO) 函 数 ， 将 数组 中 的 | 


当前 元 素 写 入 到 数组 $results 中 。 | 
最 后 应 用 implode() 函 数 , 将 数组 $results 中 的 数据 写 入 到 一 个 字符 串 变量 $Imbs 中 ， 以 
“,” 为 分 隔 符 ， 完 成 将 指定 月 份 每 天 的 数据 写 入 到 字符 串 中 的 操作 。 
while($myrow=mysql._fetch array(Sresult)){ | 
Sresults[]=$myrow['counts']; 
Slmbs=implode(",", $results); 
} 
?> 


4.7.4 动态 生成 柱 形 图 分 析 网 站 访问 量 


动态 生成 柱 形 图 分 析 网 站 访问 量 ， 主 要 是 应 用 Jpgraph 类 库 ， 根 据 从 数据 库 中 读 取 的 | 
数据 生成 柱 形 图 ， 完 成 对 网 站 访问 量 的 分 析 。 其 前 提 是 必须 在 项 目 中 载 入 Jpgraph 类 库 。 | 
由 于 其 属于 第 三 方 组 件 ， 在 配 书 光盘 的 源 代码 中 没有 提供 ， 所 以 需要 读者 自行 下 载 ， 并 且 | | 
将 下 载 的 src 文件 夹 复制 到 项 目 mr\4\05 文件 夹 下 ， 然后 才 可 以 完成 动态 生成 柱 形 图 分析 网 | 
站 访问 量 的 操作 。 
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购物 车 模块 
( 名 自 学 视频 、 源 程序 ; 配套 资源 vnA5\) 


在 网 上 购物 时 ， 经 常会 用 到 购物 车 。 这 是 一 种 用 于 临时 存储 商品 的 工具 
非常 方便 、 实 用 。 有 了 它 ， 再 也 不 用 担心 一 次 购买 多 个 商品 时 要 进行 多 次 提交 
结算 的 操作 ， 可 以 将 其 逐一 放 入 购物 车 中 ， 等 选 购 完 所 有 商品 之 后 ， 再 一 起 进 
行 结算 。 购 物 车 是 电子 商务 类 网 站 中 一 项 必 不 可 少 的 功能 ， 基 实现 的 方法 也 很 
多 ， 这 里 将 向 广大 读者 介绍 一 种 最 为 常用 的 购物 车 一 一 SESSION 购物 车 。 本 章 
将 详细 地 讲解 SESSION 购物 车 的 创建 和 使 用 方法 。 通 过 本 章 的 学 习 ， 读 者 能 够 
学 到 以 下 内 容 。 


Ph 
Ph 


SESSION 购物 车 的 创建 

更 改 购物 车 中 的 商品 数量 
删除 购物 车 中 的 商品 

统计 购物 车 中 的 商品 金额 

向 购物 车 中 添加 商品 

清空 购物 车 

通过 PDO 操作 MySQL 数据 库 
Smarty 模板 实现 网 页 动静 分 离 
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5.1 购物 车 模块 概述 


5.1.1 功能 概述 | 好 


这 里 的 购物 车 模块 是 以 电子 商务 网 站 为 大 背景 进行 开发 的 , 不仅 实现 了 购物 车 本 身 应 | Note 
该 具备 的 功能 ， 如 添加 商品 、 删 除 商品 、 更 改 商 品 数量 、 商 品 金 额 小 计 、 商 品 金额 总 计 和 | 

清空 购物 车 等 ， 而 且 还 包括 了 电子 商务 网 站 的 其 他 功能 ， 如 用 户 注册 、 用 户 登录 、 订单 查 | 

询 和 商品 分 类 等 。 其 功能 结构 如 图 5.1 所 示 。 


[购物 车 模块 ”| 


图 5.1 购物 车 模块 功能 结构 图 
5.1.2 ”购物 车 操作 流程 


购物 车 的 操作 流程 : 首先 ， 登 录 到 网 站 中 浏览 商品 ， 然 后 ， 购 买 指定 的 商品 ， 进 入 购 | 
物 车 页 面 中 ， 在 该 页 面 可 以 实现 很 多 操作 ， 包 括 更 改 商品 数量 、 删 除 商品 、 清 空 购物 车 、 | 
继续 购物 等 ， 最 后 ， 依 次 执行 填写 收 货 人 信息 、 生 成 订单 、 订 单打 印 ， 以 及 提交 订单 等 操 | 
作 。 其 操作 流程 如 图 5.2 所 示 。 


开始 
章 
购物 车 列表 
查看 购物 车 添加 商品 物 除 商 品 修改 商品 数量 | | | | 清空 购物 车 收 绢 台 结账 
| ! ! | 
(ss ) 湛 加 商品 和 (5 ) 


图 5.2 购物 车 操作 流程 | 
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和 认识 ， 


5.1.3 ”程序 预览 


VAI 


购物 车 模块 只 是 电子 商务 网 站 中 的 一 个 子 功能 , 为 了 让 读者 对 本 模块 有 个 初步 的 了 解 
下 面 列 出 几 个 典型 功能 的 页 面 。 

明日 购物 商城 网 站 主页 如 图 5.3 所 示 ， 其 中 包括 部 分 最 新 商品 、 热 门 商品 、 推 荐 商品 
会 员 登 录 窗 口 。 


以 及 网 站 的 最 新 公告 


购物 车 页 面 如 图 


图 5.3 明日 购物 商城 网 站 主页 
推荐 商品 展示 页 面 如 图 5.4 所 示 ， 以 分 页 的 形式 展示 了 网 站 的 所 有 推荐 商品 。 


推 逢 两 局) 

商品 各 次: 自生 车 
商品 扣 :超时 
风流 量 :1 

市 场 从 :3298 元 

会 员 价格 : 38 2 元 


再 品 各 答 : 要 码 要 机 
商品 品牌 : 住 和 
制订 光量: 1 

市 场 从:1583 元 
会 员 伏 格 : 1599 元 


商品 名 称 : 宏 话 旦 
商品 品 fe :人 
利多 量 : 1 

十 扬 惟 : 吉 33 元 


会 员 价格 : 4003 元 


向 :2 商品 刘 号; 000105 
商品 产地: 中国 
馆 信 光 县:1 
上 市 日 则 ;2000503 

考评 洁 。 同 买 
商号:1 十 时 呈 ;0032017 
商品 产地 :中国 
铀 千 量 : 1 
上 市 昌 阳 ;200050: 

本 
商 8 中 :2 商品 型 号 ; 0020001 
商品 产地: 中国 
销售 粘 量 : 1 
上 而 日 期 :2909.05.03 

要 在 并 第 。 巾 买 


严 品 5 个 呈 页 3 个 第 1 古典 2 页 首页 上 一 万 下 -页 尼 页 


图 5.4 推荐 商品 展示 页 面 


5.5 所 示 ， 


展示 了 会 员 在 本 网 站 购买 的 商品 ， 
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并 提供 了 购物 车 的 各 种 


5 自行 车 388 349.2 9 1047, 
厂 孝王 相机 后 1888 1699.2 9 B156 
[] E32 2666 2399.4 9 395.4 
厂 ”这 匠 影院 [mm 4888 4599.2 4399. 反 
全 迁 5 造 “同志 是 共计 ，16342.2 元 食 扩 | 


图 5.5 购物 车 页 面 ee 
5.2 数据 库 设计 
无 论 是 什么 系统 软件 ， 其 最 根本 的 功能 就 是 对 数据 的 操作 与 使 用 。 因 此 ， 人 
好 数据 库 的 分 析 、 设 计 与 实现 ， 然 后 才能 实现 对 应 的 功能 模块 。 
5.2.1 创建 数据 库 
首先 来 看 一 下 电子 商务 平台 所 使 用 的 数据 库 与 数据 表 的 情况 ， 如 图 5.6 所 示 。 


加 服务 器 : localhost 》 加 数据 库 : db_business 
一- EL 二 查询 于 i 号 县 作 es 一 一 车 


Te 国画 加 全 轩 太 -rr A 
也 cass 加 只 加 六 恒 入 全 WISAM un8_ unicode ol 。 商品 类别 表 

也 commo 贺 上 加 大便 XX 4 MSAW ulia_unicode_o 商品 信息 去 

了 fom 加 本国 大便 XX 7 MISAW ws_unicode_d 商品 订单 

也 nks 加 办 加 大便 2 MSAW we_unicode_d 友情 镑 撞 末 

mopnion 可 办 本 钾 X 9 MYSAM us_unicoge_q 商品 留 本 大 

tb _public 医 甘 哇 XX 4 MISAM ule_uricode_d 。 商品 公 告 来 

tmser 加 办 网 太 XxX MISAW ute_unicode_o 。 多 员 信 息 表 1 
8 个 来 总 计 48 MyISAM utf8_gonoral_ci 


图 5.6 电子 商务 数据 库 与 数据 表 
5.2.2 ”创建 数据 表 


下 面 来 看 各 个 数据 表 的 结构 和 字段 说 明 。 
回 tb_user (会 员 信息 表 ) 
会 员 信息 表 主 要 用 于 存储 注册 会 员 信息 ， 其 结构 如 图 5.7 所 示 。 


本 localhost ， 局 数据 库 : db_business ， 国 表 :tb_user 
区 理 


属性 Nuf 默认 格外 斋 图 上 
香 aamait [一下 二 | 
ute_unicode_ci 月 户 名 称 


要 
ge_unicode 亚 
ua_unicode_ 5 要 
utB_unicode oi 理 
Bunicoge 9 天 
utB_unicoge_9 再 
ugB_unicode 9 至 
unicode 5 至 应 机 到 
Unicode 5 到 
HEB_unicode 否 

机 
E3 
否 
再 
是 


出 


Note 


tb_commo〔 商 品 信 息 表 ) 
商品 信息 表 主 要 用 于 存储 商品 的 相关 信息 ， 其 结构 如 图 5.8 所 示 。 


加 服务 器 : localhost ， 导 数据库: db_business ， 国 表 :tb_commo 


字 队 类型 至 理 属性 Mul 。 要 认 开外 凋 明 
a nts) 至 aulo_nrement 。 自 未 总 
vame 。 Yarchar(50) ut8_unicode_o 要 商品 才 种 
Mes vacha(200) uf3 unicode d 至 picsinuljpg PE 
mmedumiet ut unicode 百 商人 名 
| 沭 加 Ht 间 

Ha_inicode 否 商品 产地 

ua_unicode 二 商品 型 号 

ua_unicode_di 本 商品 到 

UEB_unicode_ E3 两 品名 称 

二 1 两 床 和 

晋 0 EE 

否 市 场 们 格 

导 会 员 们 格 

百 9 所 宣 

村 下 是 本 新品 

本 四 朵 是 否 推 荐 


图 5.8 商品 信息 表 结 构 
此 外 ， 还 有 管理 员 表 、 商 品类 别 表 、 商 品 订 单 表 、 商 品 公 


表 、 用 户 信息 表 、 友 情 链 


全 
， 接 表 和 商品 留言 表 等 ， 限 于 篇 幅 ， 这 里 不 再 介绍 ， 具 体 参见 配套 资源 中 的 数据 库 文件 。 


5.3 首页 设计 


5.3.1 首页 概述 


首页 一 般 没有 多 少 实质 的 技术 , 主要 是 加 载 一 些 功能 模块 ， 如 登录 模块 、 导 航 栏 模块 、 


公告 栏 模块 等 ， 使 浏览 者 能 够 了 解 网 站 内 容 和 特点 。 首 页 设计 的 关键 是 要 合理 地 对 页 面 进 
| 行 布局 ， 既 要 尽 可 能 地 将 重点 模块 显示 出 来 ， 同 时 又 不 能 因为 页 面 凌乱 无 序 ， 而 让 浏览 者 
无 所 适 从 、 产 生 反 感 。 本 模块 首页 的 设计 效果 如 图 5.9 所 示 。 


图 5.9 首页 设计 效果 
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5.3.2 首页 实现 过 程 


(1) 创建 index.php 动态 页 。 在 index.php 动态 页 中 ， 应 用 include_once0 语 句 包 含 相 | 
应 的 文件 ;应 用 switch 语句 ， 以 超 链接 中 参数 page 传递 的 值 为 条 件 进行 判断 ， 实 现在 不 | 


同 页 面 之 间 跳 转 。index.php 的 关键 代码 如 下 : | 到] 
01 <?php Note 
02 session start(); | 
03 header ( "Content-type: text/html; charset=UTF-8"); // 设 置 文件 编码 格式 | 
04 require("system/system.inc.php"); // 包 含 配置 文件 | 
05 iflisset($_GET["page"]){ | 
06 $page=$_GET["page"]: | 
07 Jelse{ | 
08 Spage=""; | 
09 } | 


10 include_once("login.php"); 
11 include_once("public.php"); 
12 include_once("links.php"); 
13 switch($page){ 


14 case "hyzx": | 
15 include_once "member.php"; | 
16 S$smarty->assign(admin phtmlwmembertpl): /将 PHP 脚本 文件 对 应 的 模板 文件 ， 
名 称 赋 给 模板 变量 | 
17 break:; | 
18 case 'allpub': | 
19 include_once 'allpub.php'; | 
20 Ssmarty->assign('admin_phtml',allpub.tp1); /将 PHP 脚本 文件 对 应 的 模板 文件 | 
名 称 赋 给 模板 变量 | 
21 break: | 
22 case ‘nom': | 
23 include_once 'allnom .php': | 
24 Ssmarty->assign('admin_phtml','allnom.tpl); /将 PHP 脚本 文件 对 应 的 模板 文件 | 
名 称 赋 给 模板 变量 | 
25 break: | 
26 case ‘new': | 
27 include_once 'allnew.php'; | 
28 S$smarty->assign(admin phtml',allnewtpl); /将 PHP 脚本 文件 对 应 的 模板 文件 | 
名 称 赋 给 模板 变量 | 
29 break; | 
30 case hot : | 
31 include_once 'allhotphp': | 
3 $smarty->assign('admin_phtml','allhot.tp1); /将 PHP 脚本 文件 对 应 的 模板 文件 | 
名 称 赋 给 模板 变量 | 
33 break: | 
34 case 'shopcar': | 
35 include_once 'myshopcar.php'; | 
36 S$smarty->assign(admin phtmlmyshopcartpl):// 将 PHP 脚本 文件 对 应 的 模板 文 | 
件 名 称 赋 给 模板 变量 | 
Ed break: ! 
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| 38 case 'settle': 
| 39 include_once 'settle.php': 
| 40 S$smarty->assign(admin phtml'settle tpl): /将 PHP 脚本 文件 对 应 的 模板 文件 
， 名称 赋 给 模板 变量 
| 41 break; 
42 case 'queryform': 


- 


| 件 名 称 赋 给 模板 变量 


45 


48 


”名称 赋 给 模板 变 


49 
50 
51 
52 


include_once 'queryform.php'; 
$smarty->assign('admin_phtml','queryform.tp1);// 将 PHP 脚本 文件 对 应 的 模板 文 


break; 
default: 
include_once newhot.php'; 
S$smarty->assign(admin _ phtml',newhottpl); /将 PHP 脚本 文件 对 应 的 模板 文件 


break; 


} 
S$smarty->display("index.tp1"); /指定 模板 页 
> 


| (2) 创建 system\templates\index.tpl 模板 页 。 在 模板 文件 index.tpl 中 应 用 Smarty 的 
| include 标签 调用 不 同 的 模板 文件 ， 生 成 静态 页 面 。 其 关键 代码 如 下 : 


01 
02 


<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td colspan="2"> {include file='top.tpl}</td> 
</tr> 
<tr> 
<td width="216" align="left" valign="top"> 
{include file='login.tpl'} 
{include file='public.tpl'} 
{include file='"links.tpl'} 
</td> 
<td width="634" height="700" align="center" valign="top"> 
{include file='search.tpl'} 
<!-- 载 入 模板 文件 -->{include file=$admin_phtml}</td> 
</tr> 
</table> 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td> {include file='buttom tpl}</td> 
</tr> 
</table> 


| 说明， 


”本 模块 的 功能 较 多 ， 结 构 比 较 复杂 ， 对 于 初学 者 来 说 学 起 来 可 能 会 比较 困难 。 所 以 ， 
”本 书 将 系统 中 的 各 个 功能 模块 所 涉及 的 文件 ( 如 PHP、TPL、CSS、JS 等 ) 尽 可 能 都 单独 


| 实现 。 读者 在 学 习 其 中 某 个 模块 时 ， 可 以 将 相关 的 文件 统一 放 到 同一 个 目录 下 单独 测试 。 
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5.4 ”登录 模块 设计 


5.4.1 登录 模块 概述 | 


登录 模块 是 会 员 与 系统 进行 交互 的 窗口 。 匿 名 用 户 虽然 也 可 以 访问 本 网 站 ， 但 只 能 进 | 
行 浏览 、 查 询 等 简单 操作 ， 而 会 员 则 可 以 购买 商品 ， 并 且 能 享受 超 低 价 格 。 | 
用 户 注册 和 用 户 登 录 两 部 分 ， 其 运行 效果 如 图 5.10 所 示 。 


Note 


图 5.10 登录 模块 运行 效果 
5.4.2 用户 注 册 


用 户 注册 的 主要 功能 是 注册 新 会 员 。 如 果 信 息 输入 完整 而 且 符合 要 求 ， 则 系统 会 将 该 | 
户 信 息 保 存 到 数据 库 中 ， 否 则 显示 错误 原因 ， 以 便 用 户 改正 。 用 户 注册 页 面 的 运行 效果 | 
如 图 5.11 所 示 。 


BE Wd memer Epo 
je] rep /on ten 1 50/ soF Oureeer hp 


新 用 户 注册 


用 记名， 区 ， 用户 名 入 右 用 1 
注册 窗 码 ， esse ，。 三 和 正确 

确认 宇 码 ， [wwwew ， 给 入 正确 

窗 保 问题 ， 困 和 和 。 狂人 正确 

安保 符 军 ， [关机 ， 给 和 正确 
真实 址 名， RE  。 三 和 正确 

移动 电话 ， [EU ， 联系 电话 只 能 由 雪 字 
QQ3 ee 纺 入 正确 


图 5.11 用 户 注册 页 面 


具体 实现 过 程 如 下 。 | 
(1) 创建 register.tpl 模板 文件 ， 编 写 用 户 注 册页 面 。 其 中 包含 两 个 JS 脚本 文件 ， 即 | 
createxmlhttpjs 和 checkjs。 其 中 ，createxmlhttpjjs 是 Ajax 的 实例 化 文件 ， 而 checkjs 是 对 ， 
用 户 注册 信息 进行 验证 ， 并 且 返 回 验 证 结果 。 | 
(2) 创建 registerphp 动态 PHP 文件 ， 加 载 模板 。registerphp 文件 的 代码 如 下 : 
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上 SS 

01 <?php 

| 02 header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
| 03 require("system/system.inc.php"); // 包 含 配 置 文件 


04 $smarty->assign('title',' 新 用 户 注 册 "); 
| 05 $smarty->display(registertpl); 
| 06 ?> 


(3) 创建 reg_chk php 文件 , 获取 表单 中 提交 的 数据 , 将 数据 存储 到 指定 的 数据 表 中 。 
| reg_chk.php 的 代码 如 下 : 
01 <?php 
| 02 session start(); 
| 03 header ( "Content-type: text/html; charset=UTF-8"); /设置 文件 编码 格式 
| 04 require("system/system.inc.php"); // 包 含 配置 文件 
| 05 Sname = $_ POST['name']; 
| 06 S$password = md5($_ POST['pwd1"]); 
| 07 $question =$_ POST['question']:; 
| 08 Sanswer = $_ POST['answer']: 
| 09 Srealname =$_ POST[realname']; 
| 10 $card=$_ POST['card']: 
| 11 Stel = $_POST['tel']: 
| 12 S$phone = $_POST['phone']: 
| 13 S$Email = $_ POST['email]; 
| 14 $QQ = $_POST['qq']; 
| 15 S$code = $_POST['code']; 
| 16 Saddress = $_POST['address']: 
| 17 $addtime = date("Y-m-d H:i:s"); 
| 18 $sql="insertinto 


tb_user(name,password,question,answer,realname,card,tel.phone,Email,QQ,code,address, 

19 addtime,isfreeze,shopping,consume)" : 

20 $sql .= " values (‘$name', '$password', '$question', '$answer', '$realname', '$card', '$tel', 
'$phone', '$Email', 

21 '$QQ', '$code', '$address','$addtime','0',",'00.00")"; 


| 22 Srst= $admindb->ExecSQL($sql,$conn); 
| 23 ifSrsb{ 
| 24 $_SESSION['member'] = $name; 
25 echo "<script>top.opener.location.reload();alert(' 注 册 成 功 "):window.close(); 
| </script>"; 
| 26 jelsef{ 
| 27 echo '<script>alert(\' 添 加 失败 \');history.back;</script>"'; 
28 } 
29 ea 


i 
Er 


| (4) 在 system\templates\login.tpl 模板 页 中 ,创建 “用 户 注 册 ” 超 链接 。 当 用 户 单 
前台 的 莘 于 至 按钮 时 ， 系 统 会 调用 JS 的 onclick 事件 ， 弹 出 注册 窗口 。 其 代码 如 下 : 


| <a href="#" id="login" onclick="reg()"><img src="images/check.JPG" width="59" height="23" 
| border="0" /></a> 


这 里 用 到 的 JS 文件 为 js/loginjs， 调 用 的 函数 为 reg()。 该 函数 的 代码 如 下 : 
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01 function regO{ 


02 window.open("register.php", "_blank", "width=600,height=650",false); 
0 
5.4.3 ”用户 登录 
户 在 输入 正确 的 用 户 名 、 密 码 和 验证 码 之 后 ， 将 返回 如 图 5.12 所 示 的 信息 。 
RiPa:EE 
密码 : eewee 
验证 码 : ET] 
图 5.12 用户 成 功 登录 效果 
具体 实现 过 程 如 下 。 


(1) 创建 模板 文件 login.tpl， 完 成 用 户 登 录 表单 的 设计 。 在 该 页 面 中 ， 当 单 击 Submit 
按钮 时 ， 系 统 将 调用 lg0 函 数 对 用 户 提交 信息 进行 验证 。lg0 函 数 包含 在 js/loginijs 脚本 文 


件 内 ， 其 代码 如 下 : 


01 // JavaScript Document 
02 function lg(form){ 


if(form.name.value==""){ 
alert(' 请 输入 用 户 名 "); 
form.name .focus(); 
return false; 

} 


if(form.password.value 一 "" || form.password.value.length < 6){ 


alert(' 请 输入 正确 密码 "); 


form.password.focus(); 
Teturn false; 

} 

if(form.check.value 一 ""){ 
alert(' 请 输入 验证 码 "); 
form.check .focus(); 
return false: 

} 

if(form.check.value != form.check2.value){ 
form.check.select(); 
code(form); 
Teturn false; 

4 


var user = form.name.value: 

Var password = form.password.value; 

var Url = "chkname.php?user="+user+"&password="+password: 
xmlhttp.open("GET".urltrue): 

xmlhttp.onreadystatechange = function0{ 


二 
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| 28 if(xmlhttp.readyState 一 4){ 
| 29 var msg = xmlhttp.response Text; 
30 if(msg == "1){ 
3] alert( 用 户 名 或 密码 错误 !); 
32 form.password.select(); 
33 form.check.value = "; 
34 code(form); 
35 return false; 
36 Yif(msg —"3){ 
37 alert(" 该 用 户 被 冻结 ， 请 联系 管理 员 "); 
38 return false; 
39 }else{ 
40 alert( 欢 迎 光临 "); 
41 location.reload(); 
42 } 
43 } 
44 } 
45 xmlhttp.send(null); 
46 return false; 
47 


b 
48 /显示 验证 码 
49 function yzm(form){ 


50 var num1=Math.round(Math.random()*10000000):; 

51 var num=numl.toStringO.substr(0.4):; 

52 document.write("<img name=codeimg width=65 heigh=35 src='yzm.php?num= 
"Hn ">"); 

$3 form.check2.value=num: 

54 


} 
55 /刷新 验证 码 
56 function code(form){ 


$7 var num1=Math.round(Math.random()*10000000); 
58 var num=numl.toStringO.substr(0.4); 

59 document.codeimg.sre="yzm.php?num="+num:; 
60 form.check2.value=num: 

oly 

62 /注册 


63 function regO{ 
64 window.open("register.php", "_blank", "width=600.height=650",false); 


} 
66 ”// 找 回 密码 
67 function foundO { 
68 window.open("found.php","_blank","width=350 height=240",false):; 
co 
用 户 名 和 密码 是 在 chkname php 页 面 中 被 验证 。 
(2) 创建 用 户 信息 模板 文件 system\templates\info.tpl。 用 户 登录 成 功 后 ， 在 原 登 录 框 
位 置 将 显示 用 户 信息 。 用 户 可 以 通过 “会 员 中 心 ” 超 链接 对 自己 的 信息 进行 修改 ， 也 可 以 
单 击 “ 查 看 购物 车 ” 超 链接 查看 购物 车 商品 ， 当 用 户 离开 时 ， 可 以 单 击 “ 安 全 离开 ” 超 链 
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接 。 用 户 信息 模块 的 主要 代码 如 下 : 


01 ”<!-- 显 示 当 前 登录 用 户 名 --> 

02 欢迎 您 ， {$member} 

03 <!-- 会 员 中 心 超 链接 -> 

04 ”<a href="?page=hyzx" id="info" class="lk"> 会 员 中 心 </a> 

05 ”<!-- 查 看 购物 车 --> 

06 ”<a href="?page=shopcar" class="lk"> 查 看 购物 车 </a> 

07 ”<!-- 安 全 离开 --> 

08 <a onclick="javascript:logout()" style="cursor:hand" id="info"> 安 全 离开 </a> 


5.5 商品 展示 模块 设计 


5.5.1 商品 展示 模块 概述 


本 系统 为 用 户 提供 了 几 种 不 同 的 商品 展示 方式 ， 包 括 推荐 商品 、 最 新 商品 、 热 门 商品 | 
等 ， 能 够 使 消费 者 有 目的 地 选 购 商 品 。 每 种 展示 方式 中 均 包括 商品 的 详细 信息 显示 ,为 用 | 
户 购买 商品 提供 可 靠 的 依据 。 商 品 展示 模块 的 运行 效果 如 图 5.13 所 示 。 | 


j% 学 后 加 


mh 各 元 和 oe 
富 员 从 :349, 2 元 oo 富 员 价 ，349.2 2a ea 


遇 新 商品 ] 


d 司 等 四 皮 | 


尘 可 机 
人 人 元 5 人 元 俐 场 价 ，2566 元 市 场 价 : 4885 元 
会 员 价 ，346.2 元 。 会 员 价 ，1599.2 元 。 会 只 价 ，2300.4 元 。 会 员 价 ,4399.2 元 


EE 
图 5.13 ”商品 展示 页 面 


因为 “推荐 商品 “最 新 商品 ”和 和 “热门 商品 ”模块 的 实现 方法 和 过 程 基本 相同 , 在 | 
此 以 “推荐 商品 ”模块 为 例 进行 介绍 ， 其 他 模块 相关 代码 可 参见 配套 资源 中 的 源 程序 。 


5.5.2 ”商品 展示 模块 


在 技术 分 析 中 已 经 对 商品 展示 所 使 用 的 技术 、 方 法 作 了 概述 ， 下 面 再 来 介绍 一 下 其 具 | 
体 的 实现 过 程 。 
(1) 创建 alnom php 文件 ， 从 数据 库 中 读 取 推荐 商品 的 数据 ， 并 将 其 存储 到 模板 变 | 
量 中 。 具 体 代码 参见 5.7.6 节 。 
(2) 创建 allnom.tpl 模板 页 ， 应 用 section 语句 输出 商品 信息 ， 并 添加 相应 的 操作 按 | 
钮 或 超 链接 。 模 板 页 中 一 共有 两 个 事件 ， 查 看 商品 详细 信息 和 放 入 购物 车 。 | 
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| </font></td> 


SS 


回 ” 当 单 击 


函数 ， 


回 ” 当 单 


“查看 详情 ”按钮 时 ， 将 触发 onclick 事件 ， 并 将 调用 openshowcommo() 


同时 商品 id 会 作为 函数 的 唯一 参数 被 传递 进去 。 
ff“ 购 买 ”按钮 时 ， 同 样 会 触发 onclick 事件 ， 并 调用 buycommo() 函 数 ， 唯 


一 的 参数 也 是 商品 的 id。 
商品 模板 页 面 system\templates\allnom tpl 的 代码 如 下 : 
01 <table width="636" border="0" align="center" cellspacing="0" cellpadding="0"> 


02 
03 
&nbsp:</td> 
04 
05 
06 
07 
08 
09 
10 


<tr> 


<td height="33" align="left" valign="middle" background="images/shop 07.gif'> 


</tr> 
<tr> 


<td height="132" align="left" valign="middle"> 
{section name=nom id loop=$nomarr} 
<table width="636" height="134" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td width="145" rowspan="5" align="center" valign="middle"> 


11 <img sre="{$nomarr[nom id].pics}" alt="{$nomarr[nom id].name}" style="border: 1px solid 


#f0f0f0;" /></td> 
12 
13 
14 
15 
16 
17 
18 
19 
20 
2 
22 
3 
24 
25 


26 


<td height="35"> 商 品名 称 : {$nomarr[nom id].name}</td> 
<td width="156" height="35"> 商 品类 别 : {Snomarr[nom id].class}</td> 
<td width="157"> 商 品 型 号 : {$nomarr[nom id].model}</td> 
</tr> 
<tr> 
<td height="23"> 商 品 品牌 : {Snomarr[nom id] brand}</td> 
<td height="23" colspan="2"> 商 品 产地 : {$nomarr[nom id].area}</td> 
</tr> 
<tr> 
<td width="178" height="23"> 剩 余数 量 : {$nomarr[nom id].stocks}</td> 
<td colspan="2"> 销 售 数量 : {$nomarr[nom id].sell}</td> 
</tr> 
<tr> 
<td height="23"> 市 场 价 : <font color="red">{$Snomarr[fnom id].m _price}&nbsp; 元 


<td height="23" colspan="2"> 上 市 日 期 {$nomarr[nom id].addtime}</td> 
</tr> 
<tr> 
<td height="30"> 会 员 价 格 : <font color="#FF0000">{$nomarr[nom id].v_price} 


| &nbsp: 元 </font></td> 


30 


<td height="30" colspan="2" align="center" valign="middle"> 


31 <input id="allshow" name="allshow" type="button" value="" 

Ey class="showinfo" onclick="openshowcommo( {$nomarr[nom id].id})"/>; 
33 <input id="buy" name="buy" type="button" value="" class="buy" 

34 onclick="return buycommo( {$nomarr[nom 1d].1d})" /></td> 


35 
36 
3 
38 
3 


</t> 

</table> 

<hr style="border: 1px solid #f0f0f0;" /> 
{/section} 

</td> 
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40 </tr> 
41 <tr> 
42 <td height="25">{$rst] page}</td> | 
43 </t> | 
44 </table> | 优生 


(3) 创建 showcommojs 脚本 文件 。 当 单 击 “ 查 看 详情 ”按钮 时 ， 系 统 会 弹出 一 个 新 | 
的 页 面 ， 并 显示 商品 的 详细 信息 ， 当 单 击 “ 购 买 ” 按 钮 时 ， 该 商品 将 会 被 放 到 当前 用 户 的 其 [2 
购物 车 中 ， 如 果 没 有 登录 或 商品 已 添加 ， 则 会 提示 错误 信息 。showcommo.js 脚本 文件 的 代 | 
码 如 下 : | 
ol 放 查 看 商品 信息 函数 ， 将 打开 一 个 新 页 面 */ 


02 function openshowcommo(key){ 
03 open('showcommo.php?id="+key,'_blank',"width=560 height=300",false); 


b 
05 人 # 将 购买 商品 添加 到 购物 车 中 ， 将 在 下 节 中 讲解 所 
06 function buycommo(key){ 


08 } 


5.6 ”购物 车 模块 设计 


5.6.1 ”购物 车 模块 概述 


购物 车 是 电子 商务 平台 中 非常 关键 的 一 个 功能 模块 ， 其 主要 功能 是 保留 用 户 选择 的 商 
品 信息 ， 用 户 可 以 在 购物 车 内 设置 选 购 商品 的 数量 ， 显 示 选 购 商品 的 总 金额 ， 还 可 以 清除 | 
选择 的 全 部 商品 信息 ， 重 新 选择 商品 。 购 物 车 页 面 运行 效果 如 图 5.14 所 示 。 | 


我 的 购物 车 
会 员 价格 


5888 52992 52992 
388 3492 1047.6 


4888 43992 87984 
全 选 反选 ” 腹 聊 选择 继续 购物 去 收 殷 台 共计 : 15144 元 


图 5.14 购物 车 页 面 | 

购物 车 模块 除了 展示 用 户 购 买 的 商品 之 外 ， 还 可 实现 添加 商品 、 更 改 商 品 数量 、 删 除 

商品 和 清空 购物 车 等 操作 。 | 
5.6.2 ”购物 车 展示 


购物 车 页 面 分 为 PHP 代码 页 和 Smarty 模板 页 。 在 PHP 代码 页 中 ， 首 先 读 取 tb user | 
数据 表 中 shopping 字段 的 内 容 ， 如 果 字 段 为 空 ， 则 输出 “ 暂 无 商品 ” 如 果 数 据 库 中 有 数 | 
据 ， 则 循环 输出 数据 ， 并 将 商品 信息 保存 到 数组 中 ， 再 传 给 模板 页 。 购 物 车 页 面 | 
myshopcar.php 的 代码 如 下 : | 
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.pp 内 晶 搞 半 精 角 


01 <?php 
| 02 $select = "select id,shopping from tb_user where name =".$_ SESSION[member].""; 
| 03 Srst = $admindb->ExecSQL(S$select,$conn); // 执 行 查 询 语句 
| 04 if(grst[0O]['shopping]=—""){ // 判 断 查 询 结 果 
| 05 echo "<script>alert( 购物 车 中 暂时 没有 商品 !);window.location.href='index.php'; 


1 


07 $commarr = array(); 


| 08 ~ foreach(Srst[0]as Svalue){ /循环 读 取 内 容 
| 09 Stmpnum = explode('@',$value); 
| 10 $shopnum = count($tmpnum); // 商 品类 数 
| 11 S$sum = 0; 
| 12 foreach($tmpnum as $key => $vD){ 
| 3 $s_commo = explode(',', $v); 
| 14 $sql2 = "select id,name,m price,fold,vy_price from tb_commo"; 
| 15 $commsql = $sql2." where id=".$s_commo[0]; /定义 查询 语句 
| 16 $arr = $admindb->ExecSQL($commsql,$Sconn); /执行 查询 语句 
| 17 @$arr[0]['num'] = $s_commo[1]; 
18 @$arr[0]['total'] = $s_commo[1]*$arr[O]['v_price']: 
19 S$sum += $arr[0]['total]; 
20 $commarr[$key] = $arr[0]; // 统 计 结 果 数 据 
Dl } 
22000 
23 $smarty->assign('shoparr', $shopnum); /将 数据 赋 给 指定 的 模板 变量 


24 $smarty->assign('commarr', $commarr): 
25 $smarty->assign('sum', $sum); 
26 ?> 
购物 车 模板 页 system\templates\myshopcar.tpl 不 仅 要 负责 用 户 购买 商品 信息 的 输出 , 而 
且 还 要 提供 可 以 对 商品 进行 修改 、 删 除 等 操作 的 事件 接口 。 模 板 页 代码 如 下 : 
01 <table border="0" cellspacing="0" cellpadding="0" align="center"> 
02 <form id="myshopcar" name="myshopcar" method="post" action="#"> 


| 03 <tr> 

| 04 <td height="30" colspan="7" align="center" valign="middle" class="first"> 我 的 购物 车 </td> 

| 05 </tr> 

| 06 <tr> 

| 07 <td width="35" height="25" align="center" valign="middle" class="left">&nbsp:</td> 
| 08 <td width="100" height="25" align="center" valign="middle" class="center"> 商 品名 称 </td> 

| 09 <td width="100" height="25" align="center' valign="middle" class="center"> 购 买 数量 </td> 

| 10 <td width="100" height="25" align="center" valign="middle" class="center"> 市 场 价格 </td> 

| ji <td width="100" height="25" align="center" valign="middle" class="center"> 会 员 价格 </td> 

| 12 <td width="100" height="25" align="center" valign="middle" class="center"> 折 扣 率 </td> 

| 13 <td width="100" height="25" align="center" valign="middle" class="right"> 合 计 </td> 
| 14 </t> 

| 15 {foreach key=key item=item from=$commarr} 

| 16 <tr> 

| 17 <td height="25" align="center" valign="middle" class="left"> 


18 <input 1d="chk" name="chk[]" type="checkbox" value=" {$item.1d}"></td> 
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19 <td height="25" align="center" valign="middle" class="center"> 
20 <divid="c name{S$key}"> &nbsp; {Sitem.name}</div></td> 


21 <td height="25" align="center" valign="middle" class="center"> 

22 <input id="cnum {Skey}" name="cnum{S$key}" type="text" class="shorttxt" 
value=" {$item.num}" 

23 onkeyup="cvp({Skey}, {$item.v_price}, {$shoparr})"></td> 

24 <td height="25" align="center" valign="middle" class="center"> 

25 <div id="m price{$key}">&nbsp; {$item.m price}</div></td> 8 

26 <td height="25" align="center" valign="middle" class="center"> | 

27 <div id="v_price {$key}">&nbsp; {$item.v_price}</div></td> | 

28 <td height="25" align="center" valign="middle" class="center"> | 

29 <div id="fold{ $key}">&nbsp; {$item.fold}</div></td> | 

30 <td height="25" align="center" valign="middle" class="right"> | 

31 <div id="total{$key}">&nbsp: {$item.total}</div></td> | 

32 </tr> | 

3 {/foreach} | 

34 <tr> 

35 <td height="25" colspan="3" align="left" valign="middle"> 

36 <a href="#" onclick="return alldel(myshopcar)"> 全 选 </a> 

37 <ahref="#" onclick="return overdel(tmyshopcar):"> 反 选 </a>&nbsp:&nbsp: 

38 <input type="button" value=" 删 除 选择 " class="btn" style="border-color: #FFFFFF;" 


39 onClick = 'return del(myshopcar):> 

40 &nbsp:&nbsp:; </td> 

41 <td height="25" align="center" valign="middle"> 

42 <input id="cont" name="cont" type="button" class="btn" value=" 继 续 购 物 " 

43 onclick="return conshop(myshopcar)" /></td> 

44 <td height="25" align="center" valign="middle"> 

45 <input id="uid" name="uid" type="hidden" value="{$smarty.session.member}" > 

46 <input id="settle"” name="settle" type=" nn” class="btn”value=" 去 收银 台 "” onclick="return 
formset(form)" /></td> 


47 <td height="25" colspan="2" align="right” valign="middle"><div id='sum'> 共 计 : 
{$sum}&nbsp; 元 </div></td> 

48 </tr> 

49 </form> 

50 </table> 


5.6.3 ”更 改 商 品 数量 


对 于 新 添加 的 商品 ， 默 认 的 购买 数量 为 1， 在 购物 车 页 面 中 也 可 以 对 商品 的 数量 进行 
修改 。 当 商品 数量 发 生变 化 时 ， 商 品 的 “合计 ”金额 和 商品 总 金额 会 自动 发 生 改变 ， 该 功 
能 是 通过 触发 text 文本 域 的 onkeyup 事件 调用 cvp0) 函 数 实现 的 。cvp0 函 数 有 3 个 参数 ， 
分 别 是 商品 4、 商品 单价 和 商品 类 别 。 

在 shopcarjs 文件 中 ， 首 先 通过 商品 的 id 得 到 要 修改 商品 的 相关 表单 和 标签 属性 ， 然 
后 通过 商品 单价 和 输入 的 商品 数量 计算 该 商品 的 合计 金额 ， 接 着 使 用 for 循环 得 到 其 他 商 
品 的 合计 金额 ， 最 后 将 所 有 的 合计 金额 累加 ， 并 输出 到 购物 车 页 面 。cvp0 函 数 代 码 如 下 : 

01 function cvp(keyvprshoparr){ // 更 新 商品 数量 


a 


a Op 并 诅 模 决 精 入 


02 Varn pre = 'total'; /定义 变量 

03 var num = 'cnum'+key.toString(); /获取 传递 的 参数 值 

04 var total = n_pret+key.toString(O:; 

05 vart number = document.getfFlementById(num).value;  // 获 取 攻 值 

06 var tt] = t_number * vpr; 

07 document.getElementById(total).innerHTML = tt]; 

08 Var sm = 0; 

09 for(var i= 0; i< shoparr it+){ /循环 输出 

10 var aaa = document.getElementById(n_preti.toString0).innerText; /计算 数据 
11 sm += parseInt(aaa); 

12 } 

13 document.getElementById('sum').innerHTML = "共计 : '+sm+' 元 '; // 输 出 结果 
vo 


这 里 所 更 改 的 商品 数量 ， 并 没有 被 保存 到 数据 库 中 ， 如 果 希 望 保 存 


| 购物 ”按钮 ， 即 可 将 商品 数量 更 新 到 数据 库 中 。 
5.6.4 删除 商品 


当 对 添加 的 商品 不 满意 时 ， 可 以 对 商品 进行 删除 操作 。 操 作 流 程 为 ; 


那么 单 击 “ 继 续 


首先 选中 要 删除 


商品 前 面 的 复 选 框 ， 如 果 全 部 删除 ， 则 可 以 单 击 “ 全 选 ” 按 钮 ， 然 后 单 击 “删除 选择 ” 按 
| 钮 ， 在 弹出 的 警告 对 话 框 中 单 击 “ 确 定 ” 按 钮 ， 所 选 商品 将 被 全 部 删除 。 
| 运行 效果 如 图 5.15 所 示 。 


@ ssn xus 
厂 。 数 罗 相机 百 

呈 汽机 1 
[el 

宇 迁 反思 。 过 降 


图 5.15 删除 商品 流程 
所 有 的 删除 操作 都 是 通过 JS 脚本 文件 shopcarjs 来 实现 ， 相 关 的 函 


删除 商品 页 面 的 


数 包 括 alldel0、 


| overdel() 和 del0。alldel0 函 数 和 overdel0 函 数 实现 的 原理 比较 简单 ， 通 过 触发 onClick0 事 


件 来 改变 复 选 框 的 选中 状态 。 函 数 代 码 如 下 : 


01 /全 部 选择 /取消 
02 function alldel(form){ 


03 var leng = form.chk.length; 

04 if(leng——undefined){ 

05 if(!form.chk.checked) 

06 form.chk.checked=true; 
07 jelse{ 

08 for( vari= 0;1i< leng; i++) 
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09 { | 

10 if(!form.chk[i].checked) | 

11 form.chk[i].checked = true: | 

12 } | 

13 } | ff 
14 Teturn false; | 到 
15 } 

16 /反选 ”Note 
17 function overdel(form){ | 

18 var leng = form.chk.length; | 

19 ifleng==undefined){ | 

20 if(!form.chk.checked) | 

21 form.chk.checked=true: | 

22 else | 

23 form.chk.checked=false: | 

24 jelse{ | 

ph for( vari=0;1i< leng: i++) 

26 { 

2 if(!form.chk[i].checked) 

28 form.chk[i].checked = true; 

29 else 

30 form.chk[i].checked = false; 

31 } 

32 } 

33 return false; 

3400 


使 用 alldel0 或 overdel0 选 中 复 选 框 后 ， 即 可 调用 del0 函 数 来 实现 删除 功能 。del0 函 数 
首先 使 用 for 循环 ,将 被 选中 的 复 选 框 的 value 值 取 出 并 存 成 数组 , 然后 根据 数组 生成 URL， | 
并 使 用 xmlhttp 对 象 调用 这 个 URL， 当 处 理 完毕 后 ， 根 据 返 回 值 弹 出 提示 或 刷新 本 页 。 在 | 
shopcarjs 脚本 文件 中 ，del0 函 数 代码 如 下 : | 
01 所 删 除 记录 */ 
02 function del(form){ ! 
03 if(!window.confirm(' 是 否 要 删除 数据 ?7){ /删除 确认 | 


04 | 
05 jelse{ | 
06 var leng = form.chk.length: // 获 取 长 度 | 
07 这 leng==undefined){ | 
08 if(!form.chk.checked){ | 
09 alert(' 请 选取 要 删除 数据 !"); | 
10 jelsef | 
11 rd = form.chk.value: // 获 取 删 除 的 了 | 
12 var url 一 'delshop php?rd="+rd: /定义 执行 的 文件 | 
13 xmlhttp.open("GET",urltrue); 。 //GET 方法 传递 | 
14 xmlhttp.onreadystatechange = delnow: | 
中 xmlhttp.send(nulD): /发 送 请 求 | 
16 } | 
17 }else{ 


-2s 


.pp 内 齐 搞 半 精 角 


18 Var rd=new Array(); 

[Ey varj=0; 

20 for( vari=0;i< leng;1i++) 

21 

22 这 form.chk[i].checked){ 

23 rd0++] = form.chk[i].value; 
24 } 

25 } 

26 iflrd =—="){ 

2 alert( 请 选取 要 删除 数据 !"); 
28 }else{ 

29 var url = "delshop.php?rd="+rd; 
30 xmlhttp.open("GET",url,true); 
31 xmlhttp.onreadystatechange = delnow: 
3 xmlhttp.send(nulD); 

33 } 

34 } 

35 } 

36 return false; 

37 

38 function delnowO{ /服务 器 响应 

39 这 xmlhttp readyState 一 4){ 

40 if(xmlhttp.status 一 200){ 

41 var msg = xmlhttp.response Text: 

42 if(msg !='1){ 

43 alert( 删 除 失败 +msg); 

44 jelse{ 

45 alert( 删 除 成 功 ); 

46 location=("?page=shopcar’); 
47 } 

48 } 

49 } 

50 } 


5.6.5 ”保存 购物 车 
当 


目 户 希望 保存 更 改 后 的 商品 数量 时 ， 可 以 单 击 “ 继 续 购 物 ” 按 钮 ， 将 触发 onClick() 


事件 , 调用 conshop0 函 数 保存 数据 ,该 函数 有 一 个 参数 , 就 是 当前 表单 的 名 称 。 在 conshop() 
函数 内 ， 根 据 复 选 框 和 购买 数量 文本 域 ， 生 成 两 个 数组 fst 和 snd， 分 别 保存 商品 ID 和 商 


品 数量 。 


这 里 要 注意 ， 两 个 数组 的 值 是 要 相 


互 对 应 的 ， 如 商品 1 的 ID 保存 到 fst[1] 中 ， 那 么 商 


品 1 的 数量 就 要 保存 到 snd[1] 中 , 然后 根据 这 两 个 数组 生成 一 个 URL, 使 用 xmlhttprequest 


对 象 调用 URL， 最 后 根据 回 传 信息 作 上 4 
件 中 ， 其 代码 如 下 。 


01 ” /更改 商 品 数量 
02 function conshop(form){ 


相应 的 判断 。conshop0 函 数 存 储 于 js\shopcarjs 文 
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03 Var n_pre = 'cnum'; | 
04 var lang = form.chk.length; | 
05 if(lang — undefined){ | 
06 Var fst = form.chk.value; | 
07 var snd = form.cnum0.value; | 7 
08 jelsef | 刁 认 
09 Var fst= new Array():; 

10 var snd = new Array(); Note 
11 for(vari= 0; i< lang; iH+){ | 
12 var nm =n preti.toString(); | 
13 Var stmp = document.getElementById(nm).value: 

14 这 stmp =—"||isNaN(stmp){ 

15 alert( 不 允许 为 空 、 必 须 为 数字 7); 

16 document.getElementByIdCnm).selectO; 

17 return false; 

18 } 

19 snd[i] = stmp:; 

20 var ftmp = form.chk[i].value; 

21 fst[i] = ftmp; 

22 } 

23 } 

24 Var url = 'changecar.php?fst="+fst+'&snd="+snd; // 定 义 执行 文件 

25 xmlhttp.open("GET",url,true); // 通 过 get 方法 提交 

26 xmlhttp.onreadystatechange = updatecar; 

27 xmlhttp.send(nul); // 发 送 请 求 

282000 

29 function updatecar(O){ 

30 if(xmlhttp.readyState == 4){ // 判 断 是 否 执行 成 功 

3 var msg = xmlhttp.response Text; 

32 if(msg —'1){ 

33 location="index.php'; // 页 面 跳 转 

34 }else{ 

35 alert(' 操 作 失败 "+msg); 

36 } 

3 } 

S800 


在 conshop() 函 数 中 调用 的 changecarphp 页 为 数据 处 理 页 , 该 页 将 商品 ID 和 购买 数量 
进行 重新 排列 ， 并 保存 到 shopping 字段 内 。changecarphp 页 面 代码 如 下 : | 


01 <php | 
02 session start(); 
03 header ("Content-type: text/html; charset=UTF-8"); // 设 置 文件 编码 格式 | 
04 require("system/system.inc.php"); // 包 含 配置 文件 | 
05 $sql = "select id,shopping from tb_user where name ="".$_ SESSION[member].""; 
06 S$rst = $admindb->ExecSQL($sql,$conn); 

07 S$reback = '0'; 

08 $changecar = array(); 

09 ifisset($_GET[ 芭 it]) && isset($_GET['snd])){ // 判 断 传递 参数 是 否 设置 | 
10 $fst = $_GET['fst]; // 获 取 参 数值 | 


。131 。 Lu 


了 tp 各 型 扩 志 煤 种 


11 $snd =$_ GET['snd']: 

12 Sfarr = explode(',', $fst); 

13 $sarr = explode(',',$snd); 

14 S$upcar = array(); 

15 for($i= 0; $i < count($farr); $1++){ 

16 $upcar[$i] = $farr[$1].",".$sarr[$1]; 

17 } 

18 icount(SfarD > D{ /定义 更 新 语句 
| 19 Supdate = "update tb user set shopping=".implode('@',$upcar)." where name = 
"3_SESSION[member]."": 
| 20 jelse{ 
| 21 $update = "update tb user set shopping=".$upcar[0]." where name = 
"SS_SESSION[member]."": 
| 22 } 

23 S$shop = $admindb->ExecSQL(S$update,Sconn): /执行 更 新 操作 

24 if($shop){ 

| Sreback = 1; 

26 jelse{ 

27 Sreback = 2; 

28 } 

29 } 

30 echo $reback: 

3 ?> 


5.7 技术 提炼 


| 5.7.1 数据 库 连接 、 管 理 和 分 页 类 文件 


在 数据 库 连 接 、 管 理 和 分 页 类 文件 中 定义 三 个 类 : ConDB 数据 库 连 接 类 ， 实 现 通过 


”PDO 连接 MySQL 数据 库 ，AdminDB 数据 库 管理 类 ， 使 用 PDO 类 库 中 
， 库 中 数据 的 查询 、 添 加 、 更 新 和 删除 操作 ，SepPage 分 页 类 ， 用 于 对 商城 中 的 数据 进行 分 


的 方法 执行 对 数据 


其 具体 代码 如 下 : 


页 输出 。system.class.inc.php 文件 存储 于 根 目录 下 的 system 文件 夹 下 ， 


01 <?php 
02 /数据 库 连接 类 
03 class ConnDB{ 


04 var $dbtype: 

05 Var $host:; 

06 Var $user; 

07 Var $pwd: 

08 Var $dbname: 

09 function ConnDB($dbtype,$host, $user,$pwd,$dbname){ 
10 Sthis->dbtype=$dbtype; 

11 S$this->host=$host: 

2 Sthis->user=$user; 

13 Sthis->pwd=$pwd: 


i 


// 构 造 方 法 
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14 Sthis->dbname=$dbname; | 
1 1 | 
16 function GetConnIdO{ // 实 现 数据 库 的 连接 并 返回 连接 对 象 | 
17 if($this->dbtype—"mysql" || $this->dbtype—"mssql"){ // 判 断 数据 库 类 型 | 
18 Sdsn="$this->dbtype:host=$this->host;dbname=$this->dbname"; | 
19 yelse{ | 
20 $dsn="$this->dbtype:dbname=$this->dbname"; | 
21 } | 
22 ty{ | 
23 $conn = new PDO($dsn, $this->user, $this->pwd); /初始 化 PDO 对 象 ， 就 是 创建 数据 | 
库 连 接 对 象 $pdo | 
24 S$conn->query("set names utf8"); // 设 置 编码 格式 | 
25 return $conn; | 
26 } catch (PDOException $e) { | 
27 die ("Error!: " . $e->getMessage() . "<br/>"); | 
28 } | 
29 } 
30 


} 
31 ”// 数 据 库 管理 类 
32 class AdminDB{ 


33 function ExecSQL($sqlstr, $conn){ 

34 S$sqltype=strtolower(substr(trim($sqlstr),0,6)); /对 SQL 语句 进行 截取 

3 S$rs=$conn->prepare($sqlstr):; // 准 备查 询 语句 

36 Srs->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 

3 if($sqltype=—"select"){ | 
38 Sarray=$rs->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 | 
39 if(count($array)==0 || Srs 一 false) | 
40 return false: | 
41 else | 
42 return Sarray; | 
43 }elseif ($sqltype 一 "update" || $sqltype=="insert" || $sqltype=="delete"){ | 
44 isSrs) | 
45 returm true: | 
46 else | 
47 Teturn false; | 
48 } | 
49 } | 
50 } | 
51 /分 页 类 | 
$2 class SepPage{ | 
53 Var $rs; | 
54 Var $pagesize; | 
eh Var $nowpage; | 
56 Var $array; | 
57 Var $conn; | 
58 Var $sqlstr; | 
59 function ShowData($sqlstr, $conn,$pagesize, $nowpage){ // 定 义 方法 | 
60 if(!isset($nowpage) || $nowpage—"") /判断 变量 值 是 否 为 室 | 
61 Sthis->nowpage=1; /定义 首页 从 第 一 条 记录 开始 读 取 | 
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else 

Sthis->nowpage=$nowpage; 
Sthis->pagesize=$pagesize; /定义 每 页 输出 的 记录 数 
$this->conn=$conn: // 连 接 数 据 库 返 回 的 标识 
Sthis->sqlstr=$sqlstr; // 执 行 的 查询 语句 


Soffset=($this->nowpage-1)*$this->pagesize; 

$sql=$this->sqlstr." limit $offset, $this->pagesize"; 
Sresult=$this->conn->prepare($sq]); /准备 查询 语句 
$result->execute0; // 执 行 查询 语句 ， 并 返回 结果 集 
S$this->array=$result->fetchAll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 


这 count($this->array) 一 0 || $this->array= 一 false) 
return false; 

else 
return S$this->array; 


function ShowPage($contentname,$utits,$anothersearchstr,$anothersearchstrs,$class){ 


$str=""; 

Sres=$this->conn->prepare($this->sqlstr); /准备 查询 语句 

Sres->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 
S$this->array=$res->fetchAll(PDO::FETCH_ASSOC);// 获 取 结 果 集 中 的 所 有 数据 
Srecord=count($this->array); // 统 计 记录 总 数 
S$pagecount=ceil($record/$this->pagesize); /计算 共有 几 页 


$str.=$contentname."&nbsp:".$record."&nbsp;".$utits."&nbsp; 每 页 &nbsp; 


".$this->pagesize."&nbsp;".$utits."&nbsp; ”第 &nbsp;".$this->nowpage."&nbsp; 页 / 共 


| &nbsp;".$pagecount."&nbsp; 页 "; 


86 
87 
88 
89 
页 </a>"; 
90 
91 
92 
93 
| 94 
95 
页 </a>"; 
96 
97 
98 
99 
100 
101 
102 
103 
| 104 
| 105 
106 


$str.="&nbsp;&nbsp;&nbsp:&nbsp:"; 
if($this->nowpage!=1) 
$str.="<a href=".$_SERVER['PHP_SELF']." 


?page=l1&page_type=".$anothersearchstr."&parameter2=".$anothersearchstrs." class=".$class."> 首 


else 
$str.="<font color=#555555 作 首页 </font>"; 
S$str.="&nbsp;"; 
if($this->nowpage!=1) 
S$str="<a href=".$_SERVER['PHP_SELF']."?page=".($this->nowpage-1)." 


Rpage type=".$anothersearchstr."&parameter2=".$anothersearchstrs."” class=".$class."> 上 一 


else 

$str.="<font color=#555555> 上 一 页 </font>"; 
$str="&nb: 
if($this->nowpage!=$pagecount) 

S$str.="<a hre 合 ".-$_ SERVER['PHP_ SELF']."?page=".($this->nowpage+l)." 


可 二 


&page type=".$anothersearchstr."&:parameter2=".$anothersearchstrs." class=".$class."> 下 一 页 </a>"; 


else 
$str.="<font color=#555555'> 下 一 页 </font>"; 
$str="&nbsp:": 
这 $this->nowpage!=$pagecount) 
S$str.="<a hre 合 ".$_ SERVER[PHP SELF']."?page=".$pagecount." 
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107  &page type=".$anothersearchstr"&parameter2=".$anothersearchstrs." class=".$class."> 尾 页 </a>"; 
108 else 
109 $str.="<font color= 贿 555555 吃 尾 页 </font>"; ! 
110 if(count($this->array)==0 || $this->array=—false) | 
111 Teturn "无 数据 ! "; | 1 
112 else 到 
113 return $str; 
ja } | Note 
1 | 
116 ?> ! 


5.7.2 ”Smarty 模板 配置 类 文件 


在 Smarty 模板 配置 类 文件 中 ， 对 Smarty 模板 文件 、 编 译文 件 、 配 置 文件 等 文件 路 径 
进行 设置 。system.smarty.inc.php 文件 依然 存储 于 项 目 根 目录 的 system 文件 夹 下 ， 其 代码 | 
如 下 : | 


01 <?php 

02 require("libs/Smarty.class.php"):; // 包 含 模 板 文件 

03 class SmartyProjectextends Smarty{ // 定 义 类 ， 继 承 模板 类 
04 function SmartyProjectO{ // 定 义 方法 

05 Sthis->template_dir = "./system/templates/"; // 指 定 模 板 文件 存储 位 置 
06 S$this->compile_dir = "./system/templates_c/": /指定 编译 文件 存储 位 置 
07 S$this->config_ dir ="./system/configs/"; // 指 定 配置 文件 存储 位 置 
08 S$this->cache_dir = "./system/cache/"; // 指 定 缓存 文件 存储 位 置 
09 } 

10 } 

11 a 


5.7.3 执行 类 的 实例 化 文件 


在 system.inc.php 文件 中 , 通过 require 语句 包含 system.smartyincphp 和 system.class.inc.php 
文件 ， 执 行 类 的 实例 化 操作 ， 并 定义 返回 对 象 。 完 成 数据 库 连接 类 的 实例 化 后 ， 调 用 其 | 
中 的 GetConnId() 方 法 连接 数据 库 。system_.inc.php 文件 存储 于 system 文件 夹 下 ， 其 代码 | 
如 下 : | 


01 -<?php | 
02 require("system.smarty.inc.php"); // 包 含 Smarty 配置 类 | 
03 require("system.class.inc.php"); /包含 数 据 库 连接 和 操作 类 | 
04 $connobj=new ConnDB("mysql","localhost","root","111","db_business"); /数据 库 连 接 类 实例 化 | 
05 S$conn=$connobj->GetConnId():; // 执 行 连接 操作 ， 返 回 连接 标识 | 
06 $admindb=new AdminDB(): /数据 库 操作 类 实例 化 | 
07 $seppage=new SepPage(): // 分 页 类 实例 化 | 
08 Susefun=new UseFun(); // 使 用 常用 函数 对 类 进行 实例 化 | 
09 $smarty=new SmartyProjectO: /调用 Smarty 模板 | 
10 function unhtml($params){ 

11 extract($params); 
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Stext=$content:; 

global $usefun:; 

Tetum $usefun->UnHtml($text): 
} 
$smarty->register function("unhtml","unhtml"):; /注册 模板 函数 
?> 


5.7.4 Smarty 模板 页 中 的 框架 技术 


| 在 电子 商务 网 站 首页 中 应 用 switch 语句 与 Smarty 模板 中 的 内 建 函 数 include0 设 计 一 个 
框架 页 面 ， 实 现 不 同 功能 模块 在 首页 中 的 展示 。 
| (1) switch 语句 

在 PHP 动态 文件 中 使 用 ， 根 据 超 链 接 传递 的 值 ， 包 含 不 同 的 功能 模块 。 

(2) include 标签 

在 Smarty 模板 页 中 使 用 ， 在 当前 模板 页 中 包含 其 他 模板 文件 。 其 语法 格式 如 下 : 


{include file="file name " assign=" " var=" "} 


参数 说 明 如 下 。 

回 file: 指定 包含 模板 文件 的 名 称 。 

回 assign: 指定 一 个 变量 ,保存 包含 模板 的 输出 。 

回 var: 传递 给 待 包含 模板 的 本 地 参数 ， 只 在 待 包 含 模板 中 有 效 。 


5.7.5 “Ajax 无 刷新 验证 技术 
通过 Ajax 无 刷新 技术 可 以 对 用 户 名 进行 验证 操作 ， 以 此 来 避免 用 户 在 进行 会 员 注册 
| 时 出 现 重 名 的 问题 。 其 具体 的 应 用 方法 如 下 : 


| (1) Ajax 无 刷新 技术 验证 用 户 名 是 否 被 占用 ， 调 用 的 是 check.js 脚本 中 的 chkname 
| 方法 。 其 关键 代码 如 下 : 


/*form 为 传 入 的 表单 名 称 ， 本 段 代 码 为 register 表单 */ 
function chkname(form){ 
从 如 果 name 文本 域 的 信息 为 室 ， 名 为 namel 的 div 标签 显示 如 下 信息 */ 
if(form.name.value==""){ 
namel.innerHMRIL="<font color=#F0000> 请 输入 用 户 名 ! </font>"; 
jelse{ 
放 否 则 ， 获 取 文本 域 的 值 */ 
Var user = formname.value: 
上 放生 成 url 链接 ， 将 user 的 值 传 到 chkname.php 页 进行 判断 */ 
var url = "chkname.php?user="+user:; 
许 使 用 xmlhttprequest 技术 运行 页 面 */ 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(|){ 
if(xmlhttp.readyState == 4){ 
人 # 根 据 不 同 的 返回 值 ， 在 div 标签 中 输出 不 同 信息 */ 
var msg = xmlhttp.response Text: 
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17 if(msg —'3){ | 

18 namel innerHMRI="<font color=fFF0000> 用 户 名 被 占用 ! </font>"; | 

19 Teturn false; | 

20 }else iftmsg — 2){ | 

2 namel.innerHMRI"<font color=ereen> 恭 喜 您 ， 可 以 注册 !</fonf>"; | 贪 内 
22 人 # 如 果 用 户 名 正确 ， 则 将 隐藏 域 的 值 改 为 "yes"#/ | es 
23 form.c name.value = "yes"; 

24 jelsef - Note 
25 namel.innerHMRL="<font color=green> 未 知 错误 </font>"; | 

26 } | 

27 } | 

28 } | 

29 xmlhttp.send(null); | 

30 } | 

31 } | 


在 该 函数 中 调用 了 chkname.php 页 ， 该 页 在 会 员 登 录 时 也 会 被 调用 ， 所 以 这 里 分 两 种 
情况 ， 有 密码 和 无 密码 。 无 密码 为 注册 验证 ， 当 没有 返回 结果 时 ， 说 明 该 用 户 名 可 用 ; 而 | 
有 密码 为 登录 验证 ， 和 无 密码 相反 ， 只 有 查询 记录 存在 时 ， 才 允许 登录 ， 并 将 用 户 名 和 用 | 
户 ID 存储 到 SESSION 中 。chkname.php 页 面 代 码 如 下 : 


01 <?php 

02 session_ start(); 

03 header ( "Content-type: text/html:; charset=UTF-8" ); /设置 文件 编码 格式 
04 require("system/system.inc.php"); /包含 配置 文件 


05 S$reback = '0'; 
06 $sql = "select * from tb_user where name=".$_ GET['user].""; 
07 ifisset($_GET[password'])){ 


08 $sql .=" and password =".md5s($_GET[password])."": 

109 

10 Srst = $admindb->ExecSQL($sql,$conn); 

11 if(grs){ 

12 关中 妆 录 3 

JE if($rst[0]["isfreeze’] != 0){ | 

14 Sreback= 3'; | 

15 Jelse{ | 

16 $_ SESSION[member] = $rst[0][name']: | 

17 $_SESSION['id'] = Srst[ol[sid]: | 

18 Sreback ='2'; | 

19 } | 

20 Jelse{ | 

2 Sreback = '1'; | 

22 | 

23 echo Sreback; | 

2 | 
(2) 创建 yzm.php 文件 ， 通 过 GD2 函数 库 生 成 验证 码 。 其 关键 代码 如 下 : | 

01 <?php | 

02 header ( "Content-type: text/html:; charset=UTF-8" ); // 设 置 文件 编 | 
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码 格式 
03 srand((double)microtimeO*#1000000): // 生 成 随机 数 
04 。 $im=imagecreate(60.30): /创建 画布 
05 $black=imagecolorallocate($im,0,0,0); // 定 义 背 景 
06 S$white=imagecolorallocate($im,255,255,255); // 定 义 背 景 
07 $gray=imagecolorallocate($im,200,200,200): // 定 义 背 景 
08 imagefill($im,0,0,$gray); /填充 颜色 
09 for($i=0;$i<4:8it+){ /定义 4 位 随机 数 
10 Sstr=mt rand(3,20); // 定 义 随机 字符 所 在 位 置 的 Y 坐标 
11 Ssize=mt_rand(5,8); // 定 义 随机 字符 的 字体 
12 S$authnum=substr($_GET[mum'].$i.1); // 获 取 超 链接 中 传递 的 验证 码 


13 imagestring($im, $size,(2+$1*15),$str,$authnum,imagecolorallocate($im,rand(0,130),rand 
(0,130).rand(0,130))): 
4 


} // 水 平 输出 字符 串 
15 for($i=0;$i<200;8it+){ /执行 for 循环 ， 为 验证 码 添加 模糊 背景 
16 Srandcolor=imagecolorallocate($im,rand(0,255),rand(0,255),rand(0,255)); /创建 背景 
17 imagesetpixel($im.rand(0)9%670.rand09630,$randcolon): /绘制 单一 元 素 
15 
19 。 imagepng($inm); /生成 png 图 像 
20 imagedestroy($im): /销毁 图 像 
el) ?> 
5.7.6 ”分 页 技术 


商品 展示 功能 实现 的 关键 就 是 如 何 从 数据 库 中 读 取 商 品 信息 ， 如 何 完成 数据 的 分 页 显 
示 。 首 先 ， 包 含 类 的 实例 化 文件 ， 然 后 ， 判 断 分 页 变量 page 的 值 是 否 存 在 ， 接着， 定义 
SQL 语句 ， 对 查询 结果 进行 降 寡 排列 ， 并 且 设 置 每 页 显示 3 条 记录 ; 最后， 调用 分 页 类 中 
的 方法 完成 数据 的 分 页 读 取 和 输出 。allnom.php 文件 的 关键 代码 如 下 : 


01 <?php 

02 include once("system/system.inc.php"): // 包 含 类 的 实例 化 文件 
03 iflisset($ GET['page']){ // 判 断 当前 页 变量 的 值 
04 S$page=$_GET['page']: 

05 }else{ 

06 Spage=1; 

07009 


08 Srstl=$seppage->ShowData("select * fiom {tb commo where isnom=] order by isnomid 


descvSconn 3 Spagejy/ 调 用 分 页 类 方法 


09 $smarty->assign("nomarr", $rst]1); 
10 $smarty->assign('rstl page',$seppage->ShowPage(" 产 品 "," 个 ",$_GET['‘page_type'],","a")); // 


11 $smarty->assign('title',' 推 荐 商品 "): 
12 Ea 


最 后 ， 定 义 模板 文件 allnom.tpl， 通 过 section 语句 循环 输出 存储 在 模板 变量 中 的 数据 ， 


”并 且 输 出 分 页 超 链接 。 
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5.7.7 ”购物 车 中 商品 添加 技术 | 
购物 车 功能 实现 最 关键 的 部 分 就 是 如 何 将 商品 添加 到 购物 车 ， 如 果 不 能 完成 商品 的 添 | 
I1， 那 么 购物 车 中 的 其 他 操作 也 就 没有 任何 意义 。 a 
(1) 在 商品 展示 模块 中 ， 单 击 商 品 对 应 的 “购买 ”按钮 ， 将 商品 放 到 购物 车 中 ， 并 | 食 - 
进入 到 购物 车 页 面 。 单 击 “ 购 买 ”按钮 ， 调 用 buycommo0 函 数 ， 购 买 商品 的 ID 是 该 函数 ‘Note 
的 唯一 参数 。 在 buycommo() 函 数 中 通过 xmlhttp 对 象 调用 chklogin php 文件 ， 并 根据 回 传 | 
值 作 出 相应 处 理 。buycommo0 函 数 代码 如 下 : | 
01 产 | 


02 。 * 添 加 商品 ， 同 时 检查 用 户 是 否 登 录 、 商 品 是 否 重复 等 | 
03 机 | 
04 functionbuycommo(key){ | 
05 上 根据 商品 ID， 生 成 urlt/ | 
06 var Url = "chklogin.php?key="+key:; ! 
07 人 # 使 用 xmlhttp 对 象 调用 chklogin.php 页 */ | 
08 xmlhttp.open("GET",url,true): | 
09 xmlhttp.onreadystatechange = functionO{ | 
10 if(xmlhttp.readyState == 4){ | 
11 var msg = xmlhttp.responseText; | 
12 刻 用 户 没有 登录 */ | 
13 这 msg — 2){ | 
各 alert( 请 您 先 登录 | 
15 Teturn false: | 
16 }else if(msg — '3){ | 
ee 此 商 品 已 添加 */ | 
18 alert(' 该 商品 已 添加 '); | 
19 Teturn false: | 
20 Jelse{ | 
21 此 显示 购物 车 */ | 
22 location='index.php?page=shopcar': | 
23 | 
24 } | 
25 } | 
26 xmlhttp.send(null); | 

在 chklogin.php 文件 中 将 商品 添加 到 购物 车 中 。chklogin.php 页 代码 如 下 : | 
01 <php | 
02 session start(); | 
03 header ("Content-type: text/html: charset=UTF-8"); /设置 文件 编码 格式 | 
04 require("system/system.inc.php"); // 包 含 配置 文件 | 
05 /ee | 
06 * 1 表示 添加 成 功 | 
07 2 表示 用 户 没有 登录 | 
08 * 3 表示 商品 已 添加 过 | 
09 * 4 表示 添加 时 出 现 错误 | 
10 * 5 表示 没有 商品 添加 | 
11 a 


1 Sreback = '0'; 
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| 13 这 empty($_SESSION[member])){ 


14 Sreback = 2'; 

15 }else{ 

16 Skey =$_GET['key']: 

i if($key 一 "){ 

18 Sreback = "5"; 

19 jelse{ 

20 $boo = false; 

21 $sqls = "select idshopping from tb user where name =".$ SESSION [member].""; 

22 $shopcont = $admindb->ExecSQL($sqls,$conn):; 

23 这 !empty($shopcont[0][shoppimg])){ 

24 Sarr = explode('@',$shopcont[0]['shopping'"]): 

25 foreach($arr as $value){ 

26 S$arrtmp = explode(',',$value); 

27 这 $key 一 $arrtmp[0]){ 

28 Sreback = "3'; 

29 $boo = true; 

30 break; 

31 } 

32 } 

3 if($boo == false){ 

34 $shopcont[0]['shopping'] = '"(@'".$key.',1'; 

35 $update = "update tb user set shopping=".$shopcont[0]['shopping'’]." where name = 
"$_SESSION['member]."": 

36 $shop = $admindb->ExecSQL($update,$conn); 

3 if($shop){ 

38 Sreback = 1; 

39 }else{ 

40 Sreback = "4"; 

41 } 

42 } 

43 }else{ 

44 Stmparr = $key.",1"; 

45 Supdates = "update tb user set shopping=".$tmparr." where name = 

46 Sresult = $admindb->ExecSQL($updates,$conn); 

47 if($result){ 

48 Sreback = 1; 

49 }else{ 

50 Sreback = "4'; 

51 } 

52 } 

53 } 

SA 

55 echo S$reback; 

56 ?> 


通过 分 析 上 述 代 码 可 知 ，shopping 字段 保存 的 是 购物 车 中 的 商品 信息 。 一 条 商品 信息 
包括 两 部 分 ， 即 商品 ID 和 商品 数量 ， 其 中 商品 数量 默认 为 |。 两 部 分 之 间 使 用 逗号 “,” 
分 隔 ， 如 果 添 加 多 个 商品 ， 则 每 个 商品 之 间 使 用 “@” 分 隔 。 
成 功 完 成 商品 的 添加 操作 后 ， 即 可 进入 到 购物 车 页 面 ， 执 行 其 他 的 操作 。 
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在 线 支 付 模块 
( 名 自 学 视频 、 源 程序 ; 配套 资源 mA6\) 


随 着 Internet 的 发 展 ， 电 子 商务 已 经 成 为 21 世纪 网 络 发 展 的 主流 ， 网 上 购 
物 和 网 上 支付 已 经 成 为 一 种 时 尚 。 目 前 国内 企业 和 个 人 纷纷 加 入 到 阿里 巴巴 等 
一 系列 大 型 电子 商务 网 站 中 ,通过 网 络 进行 B2B、B2C 和 C2C 的 网 上 交易 。 在 
进行 网 上 交易 的 同时 ， 付 款 方式 也 在 随 着 网 络 的 发 展 发 生 着 变化 ， 其 中 在 线 支 
付 方式 任 借 其 快捷 、 方 便 的 将 点 ， 已 经 得 到 越 来 越 多 用 户 的 认可 。 为 了 帮助 广 
大 PHP 爱好 者 开发 出 功能 更 加 完善 的 电子 商务 类 网 站 , 在 本 模块 中 将 介绍 如 何 
通过 支付 宝 实 现 电子 商务 网 站 的 在 线 支 付 功能 ， 以 及 订单 生成 和 订单 打印 技术 
的 实现 方法 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


PI 动态 生成 订单 号 

PI 订单 打印 

WI 支付 宝 支付 

WI 订单 查询 

统计 购物 车 中 的 商品 金额 

向 购物 车 中 添加 商品 

清空 购物 车 

通过 PDO 操作 MySQL 数据 库 
利用 Smarty 模板 实现 网 页 动静 分 离 


4 


VAI 


6.1 在 线 支付 模块 概述 


全 站 6.1.1 功能 概述 


JVote | 在 线 支 付 与 购物 车 的 作用 相同 ， 都 是 电子 商务 网 站 必 不 可 少 的 一 部 分 。 本 模块 在 购物 
| 车 模块 的 基础 上 ， 增 加 了 填写 收 货 人 信息 、 生 成 订单 、 订 单 预览 、 订 单打 印 和 提交 订单 等 
| 辅助 功能 ， 最 络 实现 了 支付 宝 在 线 支付 功能 。 在 线 支 付 模块 的 功能 结构 如 图 6.1 所 示 。 


购 
物 
车 
功 
能 


| 图 6.1 在 线 支付 模块 的 功能 结构 图 
6.1.2， 在 线 支付 操作 流程 
在 线 支 付 模块 的 操作 流程 与 购物 车 十 分 相似 ， 唯 一 区 别 就 是 增加 了 支付 宝 支付 的 功 


能 。 其 操作 流程 如 图 6.2 所 示 。 
| ”更 改 购物 车 
中 商品 数量 
| 
将 商品 过 中 指定 商品 
全 各 | 浏览 加 型 购 物 车 购物 车 | | 清空 购物 车 填写 收 货 
商品 人 信息 
用 户 查看 购物 车 内 商品 上 继 纺 购 物 
门 | 商品 金额 小 计 生 
成 
-| 商品 金额 总 计 和 


订单 打印 
支付 宝 支付 | 一 < 一 一 | 提交 订单 -一 
订单 预览 


图 6.2 在 线 支付 的 操作 流程 
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第 0 章 ， 在 线 支 付 模块 “一 区 j 
6.1.3 “程序 预览 | 
在 线 支付 模块 是 电子 商务 网 站 收银 台中 的 一 个 子 功能 ,为 了 让 读者 对 本 模块 有 个 初步 | 


的 了 解 和 认识 ， 下 面 列 出 几 个 典型 功能 的 页 面 。 
明日 购物 商城 网 站 填写 收 货 人 信息 的 页 面 如 图 63 所 示 ， 可 在 其 中 填写 收 货 人 姓名 、 
电话 、 联 系 地 址 等 信息 。 


图 6.3 收 货 人 信息 添加 页 面 


收 货 人 信息 填写 完毕 之 后 ， 就 是 提交 订单 ， 并 且 生 成 订单 号 ， 进 而 完成 订单 的 预览 、 
打印 以 及 在 线 支 付 等 操作 。 生成 订单 信息 的 页 面 如 图 6.4 4 所 不 。 


+ 1306200330 订单 时 间 ， 2011-05-24 092530 
:mrsoft 收 货 人 :活检 

3 130025 电话 : 13604338784 
:长春 市 

: 平邮 付款 方式 : 银行 转账 


16992 元 16992 元 


43992 元 43992 元 


总 消费 : 6098.4 元 

荧 喜 您 ! 订单 提交 成 功 。 | 

请 您 在 一 周 内 按 您 的 支付 方式 进行 汇款 :汇款 时 注 明 您 的 订单 号 :汇款 后 请 及 时 通知 我 们 。 
注意 : 请 沁 住 订单 写 。 以 便 日 后 查询 及 有 旺 间 时 使 用 。 


打印 预览 我 更 打 印 支付宝 支 人 | | 


图 6.4 生成 订单 信息 | 


在 图 6.4 所 示 页 面 中 单 击 “ 打 印 预览 ”按钮 ， 将 进入 打印 预览 页 面 ， 如 图 6.5 所 示 。 | 
单 击 “ 我 要 打印 ”按钮 ， 将 执行 订单 的 打印 操作 ， 如 图 6.6 所 示 。 
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Note 订 音 时间。 2011-05-240933:47 
获 秋 人 ， 自 敬 
训话 : 13604338784 


Er 和 全 本 类 时 用 们 
全 


注意 : 请 民 | 


自动 分 页 @) 


| “二 9 刘 


[ROW- 


图 6.6 订单 打印 


6.2 数据 库 设计 


在 线 支付 模块 与 购物 车 模块 同属 于 电子 商务 网 站 中 的 一 个 子 功能 模块 ， 二 者 使 用 的 数 
据 库 是 统一 的 ， 都 是 MySQL 数据 库 。 在 线 支付 模块 与 购物 车 模块 的 不 同 之 处 就 是 调用 的 
是 不 同 的 数据 表 。 
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第 G6 闽 在 线 支付 模 关 一 G3] | 
6.2.1 创建 数据 库 
再 来 看 一 下 电子 商务 网 站 所 使 用 的 数据 表情 况 ， 如 图 6.7 所 示 。 


轩 服务 器 :localhost ， 品 才 报 库 .db_business y 
下 车 移民 saL 户 执 案 尺 间 币 部 S 出 车 mpon 多 委 作 号 入 加 出 六 | 食 扩 | 
于 本 作 记录 妆 全 类 型 ”天 旭 EE | Bi 
am 王 轴 四 了 针 册 X 2 Wail ut uricode ong | 
cess 五 贡 加 昼 国 X 1 wsun ee 两 kg 
由 comme 本 共 加 于 屿 X 4 WSN utg_uricede_a 。 疝 品 全 电表 Note 
也 fm 可 四 加 球面 X 17 MHSAM uta_unicode_d 商品 订单 去 和 
mks 区 茵 加 六 国 X 2 wysul ut unicode_o 。 妈 民 表 | 
opmio 三 园 呈 导言 X Wa ut unicode 5 。 者 和 上 | 
mpme 下 加 四 了 弛 国 XX 4 Wisk ma mcoge_q 商品 人 和 和 | 
也 user 入 轿 癌 续 面 XX 二 WSAM utB_unicode_d 。 仿 页 信息 相 
8 个 而 号 计 SAM mg_oeneraLol 


图 6.7 电子 商务 网 站 数据 表 
6.2.2 创建 数据 表 


下 面 了 解 一 下 与 在 线 支 付 有 直接 关系 的 数据 表 的 结构 。 
回 tb_commo〈 商 品 信息 表 ) | 
商品 信息 表 主 要 用 于 存储 商品 的 相关 信息 ， 其 结构 如 图 6.8 所 示 。 | 
nn a We 


an_mcement 。 自 二 吕 


varchar200) utg_unicode_d 
manicoge_al 


piamulupg 病员 图片 


we_unicode_d 
uunicooe_ dl 
unicode of 
Unicode al 
1 商品 上 和 
了 捅 时 


打 白 于 


CE EE EE 
: 
寺 


图 6.8 商品 信息 表 结 构 
tb_form (订单 信息 表 ) 
订单 信息 表 主要 用 于 存储 订单 的 相关 信息 ， 其 结构 如 图 6.9 所 示 。 


| 
| 

| 

| 

| 

| 

| 

加 服务 器 : localhost ， 司 数 基 库 : db_business ， 国 表 :tb_form | 
a 要 Ey 下 [ | 
| 

| 

| 

! 

| 

! 


自 ao noomont i 
在 boa 
于 ba 
在 和 
于 9 
在 半价 
3 i 
在 和 
国 :去 
机 本 提名 
杰 [3 
并 了 
Ee Bl 

否 人 
可 方式 
二 CURRENT muEsrup BM 
让 


图 6.9 订单 信息 表 结 构 
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| 此 外 ， 还 有 管理 员 表 、 商 品 公告 表 、 会 员 信息 表 、 友 情 链接 表 和 商品 留言 表 等 ， 限 于 
| | 国民 这 里 不 再 一 一 介绍 ， 读 者 可 参见 配 书 光盘 中 的 数据 库 文件 。 


a 6.3 会 员 信息 模块 设计 
ET 会 员 信息 模块 概述 


用 户 登录 后 ， 即 可 看 到 会 员 信息 模块 。 在 这 里 ， 可 以 进行 查看 或 修改 个 人 信息 及 密码 、 
| 查看 购物 车 和 安全 退出 等 操作 。 本 节 只 对 会 员 信息 模块 中 的 “会 员 中心 ” 和 “安全 退出 ” 
| 进行 讲解 。 会 员 信 息 模 块 的 运行 效果 如 图 6.10 所 示 。 


会 员 编号 : 2 

会 员 名 称 : mr 

密 保 问题 : 123456 
密 保 答案 : 123456 
注册 时 间 : 2010-12.06 05:14:10 
消费 总 额 : 0 


真实 姓 名 : 潘 萤 

身份 证 号 : 220322821103653 

移动 电话 : 13604336784 

固定 电话 : 0431.84972266 
Emma : pp@sina com 
QQ 号: 3545627366 


CE CE 
图 6.10 会 员 信息 页 面 


6.3.2 会 员 中 心 
当 用 户 登 录 成 功 后 ， 将 看 到 会 员 中 心 超 链 接 。 当 单 击 该 超 链接 时 ， 会 回 传 给 当前 页 一 
个 page 值 ， 当 前 页 根据 这 个 page 值 来 载 入 memberphp 文件 。 
1. 创建 PHP 页 面 


| 在 此 首先 创建 PHP 页 面 .因为 该 模块 中 的 模板 需要 使 用 数据 库 中 的 数据 及 一 些 动态 信 
| 息 , 这 些 都 需要 在 PHP 页 中 先行 获取 及 处 理 , 然后 再 传 给 模板 页 。 会 员 中 心 页 面 的 代码 参 
| 考 6.6 节 。 

| 2 创建 模板 页 

该 模块 包括 查看 信息 模板 及 修改 密码 模板 ， 都 存储 在 membertpl 模板 文件 中 。 关 键 代 
| 码 如 下 : 


01 <link rel="stylesheet" hre 合 "css/membercss" /> 
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02 <script language="javascript" src="js/member.js"></script> | 
03 {if $check—"find"} | 
04 <p align="left">{$smarty.session.member} &gt;&gt;&gt;<a href="?page=hyzx' id="mem"> 查 | 
看 信息 </a>&gt;&gt; | 
05 <a hre 全 ?page=hyzx&action=modify' id="mem"> 修 改 密码 </a></p> | 
06 <table id="member" width="300" border="0" cellpadding="0" cellspacing="0"> | 
07 <fom id-"member" name="member" method="post” action="modify pwd chkphp" | 
onSubmit="return pwd(memben)"> ! 
08 <tr> | 
09 <td height="25" co ="2" align="center" valign="middle" id="first"><font color="#f0f0f0"> 修 | 
改 密码 </font></td> | 


10 </tr> | 
11 /省 略 了 部 分 代码 | 
12 <tr> | 
13 <td height="30" colspan="2" align="center" valign="middle"> | 
14 <input id="enter" name="enter" type="submit" value=" 修 改 " /></td> | 
15 </tr> | 


16 </form> 
17 </table> 


18 {else} 
看 信息 </a>&gt; 


20 <a href="?page=hyzx&action=modify' id="mem"> 修 改 密码 </a></p> 

21 {section name=pwd_id loop=$pwdarr} 

22 <table id='member width="500" border="0" cellpadding="0" cellspacing="0"> 

3 <fom id="member"” name="member" method="post" action="modify info chkphp" 
onSubmit="retum mem(member)"> 


19 <p align="left">{$smarty.session.member} &gt:&gt:&gt:<a href- ?page=hyzx' id="mem"> 查 | 


24 <tr> 

25 <td height="25" colspan="2" align="center" valign="middle" id="first"> 

26 <font color="#f0f0f0">{$Spwdar[pwd_id].name} 信 息 ( 不 可 更 改 信息 ) </font></td> 

27 </tr> | 
28 <tr> | 
29 <td width="25%" height="25" align="right" valign="middle" id="left"> 会 员 编号 : </td> | 
30 <td height="25" align="left" valign="middle" id="right">&nbsp; {$pwdar[pwd idlidy </td> | 
31 </t> | 
32 /省 略 了 部 分 代码 | 
33 <tr> 

34 <td height="30" colspan="2" align="center" valign="middle"> 


35 <input name="enter" type="submit" id="enter" value=" 修 改 " />&nbsp;&nbsp;&nbsp;&nbsp; 
36 <inputname="reset" type="reset" id="reset" value=" 重 置 " /></td> 


ET </tr> 
38 </form> 
39 </table> 
40 {/section} 
41 {i 

3. 创建 脚本 文件 


该 模块 的 脚本 文件 (存储 于 06jsmemberjs 中 ) 和 用 户 注册 模块 类 似 ， 都 是 对 信息 的 
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合法 性 进行 验证 ， 如 信息 是 否 为 空 、 是 否 符合 规范 等 ， 这 里 不 再 獒 述 。 


4. 创建 处 理 页 
当 信 息 验 证 通过 后 ， 系 统 将 跳 转 到 处 理 页 modify info_chkphp 进行 信息 处 理 。 信 息 处 


| 理 页 的 代码 如 下 : 


01 <?php 

02 session start(); 

03 header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 

04 require("system/system.inc.php"); // 包 含 配置 文件 

05 $sql = "update tbuser set realname=".$ POST['realname'].",card=".$ POST 
['card'].",tel=".$_POST['tel].", 

06 phone=".$ POST[phone]. "Email-"S POST['email].",QQ=".$ POST['qq'].", 
code=".$_ POST['code]."， 

07 address=".$ POST['address']." where id=".$ POST[muserid'].""; 

08 $arr = $admindb->ExecSQL($sqLSconn): 

09 if($arr) 

10 echo "<script>alert(' 修 改 成 功 "):location=('index.php");</script>"; 

11 else 

12 echo "<script>alert(' 修 改 失 败 ):history.go(-1):</script>"; 

3 ?> 


6.3.3 安全 退出 


当 用 户 需要 离开 网 站 时 ， 可 以 单 击 “ 安 全 退出 ” 超 链接 来 调用 logout0 函 数 ， 当 用 户 
确认 退出 后 ， 则 跳 转 到 logout 页 面 ， 销 毁 session 并 回 到 首页 。 安 全 退出 所 涉及 的 页 面 及 


代码 如 下 : 


01 
02 


function logoutO{ 
这 confirm(" 确 定 要 退出 登录 吗 ? "){ // 输 出 选择 框 ， 用 户 可 以 单 击 “ 确 


认 ” 或 “取消 ”按钮 


03 
logout.php 页 
04 
05 
06 


window.open('logout.php',， parent',",false); /如 果 用 户 确认 退出 ， 则 打开 


jelse 
return false; 
<?php 
session_ start(); 
header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
session_destroy(); 
echo '<script>alert(' 用 户 已 安全 退出 I\);location=(\'index.php\");</script>'; 
?> 
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6.4 收银 台 模 块 设计 


6.4.1 收银 台 模块 概述 四 会 4 


当 用 户 停止 浏览 商品 准备 结账 时 ， 可 以 单 击 购物 车 页 面 中 的 “去 收银 台 ” 按 钮 ， 将 入 | Note 


发 onClick0 事 件 ， 调 用 formset0 函 数 显示 订单 页 面 。 当 用 户 提交 订单 后 ， 系 统 将 订单 保存 | | 
到 数据 表 tb_form 中 ， 同 时 清空 购物 车 ， 并 显示 订单 信息 提醒 用 户 记 录 订单 号 。 当 货款 发 | 
出 后 ， 还 可 以 对 订单 进行 查询 。 收 银 台 页 面 的 运行 效果 如 图 6.11 所 示 。 | 


A: [而 


了 e 直 中 式 不 下滑 ! 
[长春 市 


所 


送 估 方式， 快递 
1 式 : 支 全 ~ 


图 6.11 收银 台 页 面 运行 效果 
本 节 所 涉及 的 页 面包 括 显示 订单 (formset0) 函 数 )、 填 写 订 单 (settle.php、settle.tpl)， | 
处 理 订 单 〈settle_chkphp)、 生 成 订单 〈forminfo php、forminfo.tpl) 以 及 订单 预览 打印 5 | 
部 分 。 | 


6.4.2 显示 订单 


订单 信息 提交 页 面 的 输出 由 formset0 函 数 决定 ， 它 将 商品 信息 加 以 整理 ， 通 过 open() | | 
方法 打开 settle.php 页 来 显示 订单 ,并 将 整理 后 的 商品 信息 传递 到 settle.php 文件 中 。 td 
函数 存储 于 js\shopcarjs 文件 中 ， 其 关键 代码 如 下 : 


01 function formset(form){ 
02 var uid = form.uid.value; 


03 -varn pre='cnum' // 数 量 
04 var lang = form.chk.length; 

05 if(lang 一 undefined){ 

06 var fst = form.chk.value: /商品 了 
07 var snd = formm.cnum0.value: /购买 数量 
08 jelse{ 

09 Var fst= new Array0: 

10 var snd = new Array(); 

11 for(vari= 0; 1< lang; IH){ 

1 var nm =n preti.toString(); 

13 var stmp = document.getElementById(nm).value: 
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14 这 stmp —"||isNaN(stmp){ 

15 alert( 不 允许 为 空 、 必 须 为 数字 7); 

16 document.getElementById(nm).select(); 

[a return false; 

18 } 

19 snd[i] = stmp; 

20 var ftmp = form.chk{i].value; 

21 fst[i] = fonp; 

22 l 

23 +: 

24 open('settle.php?uid="+uid+'&fst='+fst+'&snd="+snd,'_blank'’,"width=500 
height=450',false); 

25 


说 明 : 
因为 open() 方 法 使 用 了 _blank 参数 来 打开 一 个 新 的 页 面 ，session 值 传 不 过 去 ， 所 以 这 
里 使 用 隐藏 域 来 传递 用 户 名 。 


6.4.3 ”填写 订单 


settle.php 直接 将 接收 的 值 传 给 settle.tpl 模板 ， 并 载 入 settle.tpl 模板 。settle.php 页 面 的 
代码 如 下 : 


01 <2php 

02 session_ start(); 

03 header ("Content-type: text/html; charset=UTF-8"); /设置 文件 编码 格式 
04 require("system/system.inc.php"); // 包 含 配 置 文件 


05 $fst = $_GET['fst]; 

06 $snd=$_GET['snd']; 

07 Suid=$ GET['id]: 

08 $smarty->assign("title',' 收 银 台 "); 
09 $smarty->assign('fst', $fst):; 

10 $smarty->assign('snd',$snd); 

11 $smarty->assign(‘uid',$uid); 

12 $smarty->display('settle.tpl): 

13 ?> 


在 settle.tpl 模板 页 ， 首 先 载 入 settlejs 脚本 文件 ， 调 用 其 中 的 chkreginfo() 方 法 验证 表 
单 提交 的 值 是 否 符合 要 求 。 然 后 ， 创 建 表单 form_reg， 提 交 收 货 人 、 联 系 电话 、 联 系 地 址 
等 信息 , 同时 还 将 PHP 页 传递 的 变量 以 隐藏 域 的 形式 提交 到 处 理 页 settle_chk.php。settle.tpl 
模板 页 的 关键 代码 如 下 : 
01 <form name="form reg" method="post" action="settle_chk.php" onSubmit="returmn chkreginfo 
(form reg,'all)"> 


02 <tr> 
03 <td height="25" align="right" valign="middle" class="left"> 收 货 人 : </td> 
04 <td width="221" height="25" align="left" valign="middle" class="center">&nbsp; 


05 <input type="text" name="taker" size="20" onBlur="chkreginfo(form reg,0)"> 
“50's 
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06 <div id="chknew_taker" style="color:#FF0000"></div> 


07 </td> 
08 <td width="56" align="right" valign="middle" class="center"> 邮 编 : </td> 
09 <td width="207" align="left" valign="middle" class="right">&nbsp; 


10 <input type="text" name="code" size="20" onBlur="chkreginfo(form reg,1)"> 
11 <div id="chknew_code" style="color:#FF0000"></div></td> 


12 </tr> 

13 <!-- 省 略 了 部 分 代码 --> | 
14 <tr> | 
[3 <td width="116" height="25" align="right" valign="middle" class="left"> 付款 方式 : </td> | 


16 <td height="25" colspan="3" align="left" valign="middle" class="right">&nbsp;<select 
id="pay" name="pay"> 


17 <option value=" 银 行 转账 "> 银行 转账 </option> | 
18 <option value=" 邮 局 汇款 "> 邮局 汇款 </option> | 
19 <option value=" 支 付 宝 "> 支付 宝 </option> | 
20 </select> </td> | 
wl </tr> 

22 <tr> 

23 <td height="30" colspan="4" align="center" valign="middle"> 


24 <inputid="enter" name="enter" type="submit" value=" 提 交 订 单 " class="btn" /> 
25 <input id="fst" name="fst" type="hidden" value=" {$fst}" /> 

26 <input id="snd" name="snd" type="hidden" value="{$snd}" /> 

27 <input id="uid" name="uid" type="hidden" value="{$uid}" ></td> 

28 </t> 

29 </form> 


6.4.4 ”处 理 订单 


在 处 理 页 settle_chk.php 中 ,首先 获取 表单 中 提交 的 数据 ; 然后 根据 用 户 提交 的 商品 信 
息 ， 重 新 查找 数据 表 tb_commo， 并 从 数据 表 中 提取 商品 信息 ， 保 存 到 数组 中 ， 再 将 数组 
作为 一 条 记录 添加 到 tb_form 表 ; 数据 添加 成 功 的 同时 ， 处 理 页 会 根据 uid 找到 该 用 户 ， 
将 shopping 字段 清空 ， 最 后 调用 forminfo.php 页 来 显示 新 添加 的 订单 信息 。settle_chk.php 
页 的 代码 如 下 : 


01 <2php 
02 header ( "Content-type: text/html:; charset=UTF-8" ); /设置 文件 编码 格式 
03 require("system/system .inc.php"); // 包 含 配置 文件 


04 $sql="insert into tb_form(formid,commo _id,commo_name,commo_num.agoprice,fold,total, 
vendee,taker,address, 

05 tel,code,pay_method,del_ method,formtime,state)values("; 

06 Sformid=time():; 

07 Stmpid = explode(',',$_POSTT['fst"]); 

08 Stmpnm = explode(',',$_POST['snd'"]): 

09 Snumber = count($tmpid); 

10 Stmpna = array(); 

ll S$tmpvp=arrayO; 

12 S$tmpfd = arrayO; 

13 S$tmptt=0; 


a 
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14 if($number >1){ 


| 15 for($i= 0; $i < $number; Si++){ 
| 16 $tmpsql = "select name,v_price,fold from tb_commo where id =".$tmpid[$1].""; 
| 到 Stmprst = $admindb->ExecSQL($tmpsql,$conn); 
18 Stmpna[$1i] = $tmprst[0]['name']; 
19 Stmpvp[$i] = Stmprst[0][v_price]; 
20 Stmpfd[$i] = $tmprst[0]['fold"]: 
| 21 S$tmptt += $tmprst[0][v_price] * $tmpnm[$1]; 
| 22 @$tmpsell = $tmprst[0]['sell] + 1; 
| 23 $addsql = "update tb_commo set sell = ".$tmpsell." where id =".$tmpid[$1].""; 
| 24 $addrst = $admindb->ExecSQL($addsql.Sconn): 
| 25 } 
| 26 $sq] $_POST['fst].",".implode(',',$tmpna).",".$_POST['snd']." 


| mimplode( ,Stmpvp).", 
| 27 mimplode(,,$tmpfd).",".$tmptt.",".$_POST['uid']."™"; 
28 }else if($number 一 1){ 


29 S$tmpsql = "select name,v_price,fold from tb_commo where id =".$tmpid[0].""; 
30 S$tmprst = $admindb->ExecSQL($tmpsql,$conn):; 

31 S$tmptt= $tmprst[0]['v_price'] * $tmpnm[0]: 

32 @$tmpsell = $tmprst[0]['sell] + 1; 

33 $addsql = "update tb_commo set sell = ".$tmpsell." where id = ".$tmpid[0].""; 
34 $addrst = $admindb->ExecSQL($addsql,$conn); 

35 $sql.="".$formid.",".$_ POST['fst].",".$tmprst[0]['name'].",".$_POST['snd'].", 


| ".$tmprst[0]['v_price'].", 
| 36 "S$tmprst[O]['fold].",".$tmptt.™,".$ POST['uid]."™"; 


3 jelsef 

38 echo 'error': 
39 exit(); 

40 > 


| 41 $sql=",".$ POST['taker].",".$ POST['address].",".$ POST['tel].",".$_POST['code'].", 
| "8_POST['pay'].", 

42 "mS$ POST['del].",".date("Y-m-d H:i:s").".0)"; 

43 S$InsertSQL = $admindb->ExecSQL($sql,$conn); 

44 iflfalse = $InsertSQL){ 


| 45 echo "<script>alert(' 购 买 失 败 ");history.back;</script>"; 
| 46 jelse{ 
| 47 $updsql = "update tb_user set consume='".$tmptt",shopping=" where name = ".$_ POST 
| [mid] ww: 
| 48 Supdrst = Sadmindb->ExecSQL(SupdsqlSconm): 
49 echo "<script>top.opener.location.reload();</script>"; 
| 50 echo "<script>open('forminfo.php?fid=$formid',' blank','width=750 
height=650",false);window.close();</script>"; 
| Si 
$2 ?> 
| 6.4.5 生成 订单 


艺 
性 


E 成 的 关键 是 


订单 处 理 成 功 之 后 ， 将 跳 转 到 forminfo php 页 输出 生成 的 订单 。 订 证 
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forminfo.php 和 forminfo tpl。 
在 forminfo.php 文件 中 ， 根 据 超 链接 传递 的 订单 号 查询 订单 表 中 的 数据 ， 并 将 查询 结 
果 赋 给 指定 的 模板 变量 ， 最 终 指定 模板 页 forminfo.tpl。 其 关键 代码 如 下 : 


01 
02 


<?php 

header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
include_once("system/system.inc.php"); // 包 含 类 的 实例 化 文件 
$ddnumber=$_GET['fid']: // 获 取 订 单 号 

$sql = "select * from tb_form where formid=".$_GET['fid].""; // 定 义 SQL 语句 
Sformarr = $admindb->ExecSQL($sql,$conn); // 执 行 SQL 语句 
Scommname = explode(,,$formarr[0]['commo_name]); // 读 取 数 据 库 中 数据 
$commnumber = explode(',,$formarr[0]['commo_num']): 

$commagoprice = explode(',', $formarr[0]["agoprice']); // 读 取 数 据 库 中 数据 
$commfold = explode(',', $formarr[0]['fold"]); 

$smarty->assign('formarr', $formarr[0]); // 将 数据 赋 给 模板 变量 


$smarty->assign('commname',$commname); 
$smarty->assign('commnumber',$commnumber); 
$smarty->assign('commagoprice',$commagoprice); 
$smarty->assign('commfold',$commfold): 
if($formarr[0]["del_method"]==" 平 邮 "){ 
S$yprice="200"; 
}else{ 
$yprice="300": 


} 

-…-// 省 略 部 分 代码 

$smarty->assign('title',' 订 单 提交 成 功 "); 

$smarty->display(forminfo .tpl); /指定 模板 页 


?> 


在 forminfo.tpl 模板 页 中 , 获取 模板 变量 传递 的 值 , 完成 订单 信息 的 输出 , 并 且 创建 “我 
要 打印 “打印 预 览 ” 按 钮 和 “支付 宝 支 付 ” 超 链接 。 其 关键 代码 如 下 : 


01 
02 


15 
16 
&nbsp; 元 </td> 


<table width="650" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td height="30" colspan="5" align="center" valign="middle" class="first"> 订 单 内 容 </td> 
</tr> 
<tr> 
<td width="100" height="25" align="center" valign="middle" class="left"> 商 品名 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 数 量 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 价 格 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 价 格 折 率 </td> 
<td width="100" height="25" align="center" valign="middle" class="right"> 合 计 </td> 
</tr> 
{foreach key=key item=item from=$commname} 
<tr> 
<td height="25" align="center" valign="middle" class="left">{$item}</td> 
<td height="25" align="center" valign="middle" class="center">{$commnumber[$key]}</td> 
<td height="25" align="center" valign="middle" class="center">{$commagoprice[$key]} 
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17 <td align="center" valign="middle" class="center">{$commfold[$key]} &nbsp: 折 </td> 
18 <td align="center” valign="middle" ="right">{$commagoprice[$key]*$commnumber 
[$key]} &nbsp; 元 </td> 
| 19 </tr> 
天 | 20  {/foreach} 
NA | 功用 二 ,过 
| 22 <td col ="$" height="25"”align="right”valign="middle"> 总 消费 {$formarr.total} 
&nbsp; 元 </td> 
| 23 </tr> 
24 </table> 


25 。 <input type="button" value=" 打 印 预览 " onclick="printview(0" class="btn" />&nbsp;&nbsp; 
26 <input type="button" value=" 我 要 打印 " onclick="prints0" class="btn" /> &nbsp:&nbsp; 

27 <!--<input type="button" value=" 我 要 打印 " onclick="window.printO" class="btn" />--> 

28 ”<!-- 支 付 宝 支 付 的 接口 操作 ， 提 交 的 数据 --> 

29 <a href="{$link}"> 支 付 宝 支付 </a> 

30 <!- > 


”说 明 : 
。 本 模块 中 的 订单 打印 、 打 印 预览 和 支付 宝 支付 功能 都 是 在 forminfo.php 和 fonminfo.tpl 
文件 中 完成 的 ， 有 关 这 些 内 容 将 在 后 文中 进行 详细 介绍 。 


6.4.6 订单 预览 、 打 印 


在 线 支付 模块 中 应 用 WebBrowser 技 术 实 现 订 单 的 打印 和 预览 功能 , 运行 效果 如 图 6.12 
| 和 图 6.13 所 示 。 


届 印 @.， | 天 | * 了 由 站 站 一 “| 东区 [ge ED 


tp//i92 168 1  - 可 和 区 要 功 


i 


页 码 @) 
输入 页 码 或 页 面 范围 。 如 ，5-12 


FD Cm | ED 条 要 和 名 支付 主 支付 


图 6.12 订单 打印 图 6.13 打印 预览 
| 本 模块 中 的 打印 和 预览 功能 在 forminfo.tpl 模板 页 中 实现 。 首 先 ， 载 入 printjs 脚本 文 
| 件 ; 然后 ， 定 义 object 标签 ， 最后， 创建 打印 预览 和 打印 的 提交 按钮 ， 并 且 通 过 onClick() 
| 函数 调用 对 应 的 方法 完成 相应 操作 。 其 关键 代码 如 下 : 


| 01 <script language="javascript" src="js/print.js"></script> 
| 02 <object ID= WebBrowserl' CLASSID='CLSID:8856F961-340A-11D0-A96B- 00C04FD705A2> 
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03 <input type="button" value=" 打 印 预览 " onclick="printview(" class="btn" /> 
04 <input type="button" value=" 我 要 打印 " onclick="prints0" class="btn" /> &nbsp:&nbsp:<! 一 
05 <input type="button" value=" 我 要 打印 " onclick="window.print()" class="btn" />--> | 
print.js 脚本 文件 中 封装 了 printview0 和 prints0 两 个 方法 ， 分 别 调用 WebBrowser 控件 | 国 内 
中 的 方法 完成 打印 预览 和 打印 操作 。 其 代码 如 下 : | 窒 一 


01 fonction printviewO{ Note 
02 document.all. WebBrowserl .ExecWB(7,1); /打印 预览 | 

03 window.close(): | 

sey 

05 function printsO{ 

06 document.all. WebBrowserl.Execwb(6,1):; /打印 

07 window.close(); 

08 } 


6.5 订单 查询 


6.5.1 订单 查询 功能 概述 


利用 订单 查询 功能 ， 可 以 根据 提交 的 订单 号 从 数据 库 中 查询 出 指定 订单 的 数据 。 其 运 
行 效果 如 图 6.14 所 示 。 


看 全 
明日 购 物 商城 es 新 年 购物 帮办 


Mttp /fwww mroook com 


首页 | 最 新 商品 1 推荐 商品 | 。 热门 商品 | 订 羊 查询 | 2011-1-1 11:46:53 
一 


了 欢迎 奈 ; mr 


会 员 中 心 新 款 彩 电 | 


汪 ， 超 值 大 甩 胸 


有 网 买 必 可 以 上 的 彩电 ， 全 部 型 kr 可 形 电 


网 站 公告 ‘ 
pp 
口 欢迎 新 老 朋友 光临 下 
五 一 大 天 宅 音 询 订单 | 
© 51001 
口 新 直 硕 目 NP: 但 询 订单 号 : |1236863343 Er 
查询 结果 
订单 号。 订 纪 会 员 。 中 全 人 订单 全 烟 。。 付 划 方式 收 吉 方式 订单 拓 态 | 
L93863443 严 全 la452 支 宝 。 送 抽 上 门 。。 未 作 处 理 


图 6.14 订单 查询 
6.5.2 ”订单 查询 功能 的 实现 | 


订单 查询 功能 由 4 个 文件 组 成 ， 分 别 是 queryform php、queryform tpl、queryformjs 和 | 

queryphp。 有 具体 实现 过 程 如 下 : 

(1) 通过 queryform php 文件 动态 PHP 文件 ， 用 于 指定 模板 页 queryform.tpl。 | 

(2) 利用 queryform.tpl 模板 页 创建 订单 查询 的 表单 。 载 入 JS 脚本 ， 通 过 JS 脚本 中 | 
。155。 \ 
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的 queryrst0 方 法 调用 queryphp 完成 订单 的 无 刷新 查询 ， 将 查询 结果 在 queryform tpl 模板 
| 页 中 输出 。queryform.tpl 模板 页 的 关键 代码 如 下 : 


01 <form id='queryform' name='queryform' method='post action='#> 


02 <tr> 
03 <td height='25' colspan='5' align='center valign="middle' class='first> 查 询 订单 </td> 
04 </tr> 
05 <tr> 
| 06 <td width='80' height='25' align='right' valign="middle' class='left'> 查 询 用 户 : </td> 
| 07 <td width='130' height='25' align=left valign="middle' class='center>&nbsp: 
| 08 <input id='name' name='mame' type='text class='txt /></td> 
| 09 
| 10 <td width='100' height='25' align='right valign="middle' class='center> 查 询 订单 号 : </td> 
| 11 <td width='130' height='25' align=left valign="middle' class='right’>&nbsp:; 
| 12 <input id='formid' name='formid' type='text class='txt /></td> 
| 13 <td width='100' align='center valign='middle' class='center> 
| 14 <input id='enter name='enter type='button' value=' 查询 ' class='btn' onclick='return 
| queryrst(queryform)/></td> 
| 15 </tr> 
| 16 <tr> 
| 17 <td height='25' colspan="5' align='center valign="middle'>&nbsp:</td> 
| 18 </tr> 


| 19 </form> 


| (3) 创建 queryformjs 脚本 文件 ， 定 义 queryrst0 和 showfm() 方 法 ， 调 用 query.php 文 


| 件 完成 订单 的 无 刷新 查询 ， 并 且 将 查询 结果 赋 给 exam。 其 代码 如 下 : 


01 // JavaScript Document 
| 02 function queryrst(form) { 


| 03 varname = form.name.value; 1/ 获取 表单 提交 的 值 

| 04 var formid = form.formid.value; 1/ 获取 表单 提交 的 值 

| 05 if(name —" && formid — "){ // 判 断 提交 的 值 是 否 为 空 
| 06 alert(' 用 户 或 订单 号 至 少 有 一 个 不 能 为 空 ); 

| 07 form.name.focus(); 

| 08 return false: 

| 09 } 

| 10 var url = "query.php?vendee="+name+"&formid="+formid; /定义 调用 的 URL 
| 11 xmlhttp.open("GET",url,true): // 执 行 调用 操作 

| 12 xmlhttp.onreadystatechange = showfim; // 回 调 函 数 

| 13 xmlhttp.sendCnulD): 1/ 执行 

| 1 

| 15 function showfmO{ 

| 16 if(xmlhttp.readyState == 4){ // 判 断 返 回 结 果 

| 2 var msg = xmlhttp.responseText: 

| 18 这 msg 一 '0){ 

| 19 exam.innerHTML = "; 

| 20 alert( 没 有 查询 结果 ); 

| 5 jelse{ 

| 3 exam.innerHIML= msg; // 将 查询 结果 赋 给 exam 
| 23 } 
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24 } 
25 


(4) 创建 queryphp 文件 ， 根 据 超 链接 传递 的 参数 值 执行 订单 查询 操作 ， 并 且 返 回 查 
询 结果 。 其 代码 如 下 : 


互 


01 -<?php 

02 header ("Content-type: text/html; charset=UTF-8"); /设置 文件 编码 格式 

03 。 include once("system/system.inc.php"); // 载 入 类 的 实例 化 文件 ， 加 载 对 象 
04 Svendee=$ GET['vendee']; // 获 取 超 链 接 传 递 的 值 

05 $formid=$ GET['formid"]; // 获 取 订单 号 


07 S$tmp = "<table width='600' border="0' align='center cellpadding="0' cellspacing='0> 

08 <tr><td height='25' colspan='7' align='center valign="middle' class='first> 查 询 结果 </td></tr> 
09 <tr><td width='100' height='25' align='center valign='middle' class='left'> 订 单 号 </td> 

10 <td width="75' height='25' align='center valign="middle' class='center> 订 货 会 员 </td> 

11 <td width="75' height='25' align='center valign="middle' class='center> 收 货 人 </td> 

12 <td width='100' height="25' align='center valign="middle' class='center> 订 单 金额 </td> | 
13 <td width='75' height='25' align='center valign="middle' class='center> 付 款 方式 </td> | 
14 <td width="75' height='25' align='center valign="middle' class='center> 收 款 方式 </td> | 
15 <td width='100' height='25' align='center valign="middle' class='right'> 订 单 状 态 </td></tr>"; | 


06 Sreback = '0'; | 


16 $sql = "select id,formid,vendee,taker,total,pay_method,del_ method,state from tb_form 
17 Where vendee = ".$vendee." or formid = ".$formid."": 


18 S$rst = $admindb->ExecSQL($sql,$conn); // 执 行 查询 语句 

19 ifsrsb{ 

20 foreach($rst as $value){ // 循 环 输出 查询 结果 

21 S$tmp .= "<tr><td height=25 align=center valign=middle class=left>".$value 


['formid']."</td> 
22 <td align=center valign=middle class=center>".$value['vendee']."</td> 
23 <td align=center valign=middle class=center>".$value['taker"]."</td> 
24 <td align=center valign=middle class=center>".$value['total]."</td> 
D5 <td align=center valign=middle class=center>".$value['pay_method']."</td> 
26 <td align=center valign=middle class=center>".$value["del method']."</td> 
2 <td align=center valign=middle class=right>"; 


28 Switch ($value['state]){ 

29 case 0: 

30 S$tmp =' 未 作 处 理 ': | 
31 break:; | 
32 case 1: | 
33 Stmp 二 ' 已 付款 '; | 
34 break: | 
35 case 2: | 
36 Stmp 二 ' 已 发 货 '; | 
37 break; | 
38 case 3: | 
39 S$tmp 三 ' 已 收 货 '; | 
40 break: | 
41 } | 
42 S$tmp = "</td></tr>"; | 
43 i | 


Ch 


VAIO 


| 44 S$tmp = "</table>"; 
| 45 Sreback = Stmp: 
| 46 } 
| 47 echo Sreback: 
内 | 48 ?> 
全 A 
pe 6.6 技术 提 浆 


6.6.1 收 货 人 信息 验证 


| 为 了 保证 网 上 交易 的 顺利 完成 ， 在 处 理 收 货 人 提交 的 信息 时 ， 必 须 对 其 填写 的 信息 进 
| 行 验证 ， 以 保证 信息 是 合理 、 有 效 的， 避免 格式 上 的 错误 出 现 。 这 里 在 验证 收 货 人 提交 的 
| 信息 时 应 用 的 是 JavaScript 脚本 和 正则 表达 式 。 验 证 收 货 人 信息 的 运行 结果 如 图 6.15 所 示 。 


ETEEEE 
地 直 : 开 大 市 

这 方式 ， 居 六 

人 表 方 式 : 支 作 宝 > 


要 i 单 


_ 本 @ imerme: | 人 Bp 模 克 要 


图 6.15 验证 收 货 人 信息 


(1) 验证 手机 号 码 格式 
| 主要 是 通过 JavaScript 脚本 的 自 定义 函数 ,调用 正则 表达 式 ， 实现 对 手机 号 码 的 验证 。 
| 编写 判断 手机 号 码 格 式 的 正则 表达 式 如 下 : 
A130d{93)8IA15Cdf9))3A 

判断 是 否 是 以 13 或 者 15 开头 ， 判 断 是 否 是 11 位 数字 。 判 断 手机 号 码 格式 是 否 正确 


| 的 自 定义 函数 的 代码 如 下 : 
| 01 function checkregtel(regteD){ 

02 Var str=regtel; 

03 var Expression=/^13(\d{9})$I^15Qd{9})$/; V 判 断 手机 号 码 的 号 段 和 格式 是 否 正确 
| 04 Var objExp=new RegExp(Expression); 
| 05 if(objExp.test(str)—true){ 
| 06 Teturn true: 

07 jelse{ 

08 Teturn false: 

09 5 

0 
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(2) 验证 收 货 人 的 其 他 信息 
验证 收 货 人 的 其 他 信息 主要 应 用 的 是 JavaScript 脚本 , 通过 JavaScript 脚本 中 的 自 定义 | 


函数 获取 表单 提交 的 数据 ， 在 自 定义 函数 中 判断 数据 是 否 合理 ， 如 果 不 合理 则 通过 div 标 
签 直接 输出 错误 提示 。 验 证 收 货 人 的 其 他 信息 的 代码 如 下 : 
01 function chkreginfo(form,mark,edit){ 


02 这 mark 一 0 || mark=—="all"){ /验证 收 货 人 
03 if(form .taker.value==""){ 
04 chknew takerinnerHTMIL=" 请 输入 收 货 人 ! "; | 
05 form.taker.style.backgroundColor="#FF0000"; | 
06 return false: ! 
07 jelsef | 
08 chknew taker.innerHTML=""; | 
09 form.taker.style.backgroundColor="#FFFFFF"; | 
10 } | 
11 } | 
12 if(mark=—=1 || mark=="all"){ // 验 证 邮编 | 
13 if(form.code.value—""){ | 
14 chknew_code.innerHTML=" 请 输入 邮编 ! " | 
5 form.code.style.backgroundColor="#FF0000"; | 
16 return false:; | 
17 j}else if(isNaN(form.code.value){ | 
18 chknew_code.innerHTML=" 邮 编 由 数字 组 成 !"; | 
19 form.code.style.backgroundColor="#FF0000"; | 
20 return false; | 
21 jelse if(form.code.value.length!=6){ | 
22 chknew_code.innerHTML=" 邮 编 由 6 位 数字 组 成 !"; | 
5 form.code.style.backgroundColor="#FF0000": | 
24 return false; | 
25 jelse{ | 
26 chknew_code.innerHTML=""; | 
2 form.code.style.backgroundColor="#FFFFFF"; | 
28 } | 
29 } | 
30 if(mark==2 || mark=="all") { // 验 证 电话 号 码 | 
31 if(form.tel.value==""){ | 
32 chknew teLinnerHTML=" 请 输入 电话 号 码 ! "; | 
33 form.tel.style.backgroundColor="#FF0000"; | 
34 Teturn 包 lse: | 
35 jelse if(!checkregtel(form.tel.value)){ | 
36 chlmew telinnerHTMIL=" 电 话 号 码 的 格式 不 正确 1 "; | 
37 form.tel.style.backgroundColor="#FF0000"; | 
38 return false; | 
39 jelse if(isNaN(form telvalue)jf | 
40 chknew telinnerHTML=" 电 话 号 码 由 数字 组 成 ! ": | 
41 form.tel.style.backgroundColor="#FF0000"; | 
42 return false; | 
43 }else{ | 
44 chlkmew tel.innerHTML=""; | 
45 form.tel.style.backgroundColor="#FFFFFF"; | 
“ods \ 
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46 上 

47 } 

48 if(mark—3 || mark—="all"){ // 验 证 联系 地 址 

49 if(form.address.value—""){ 

50 chknew_address.innerHTMIL=" 请 输入 联系 地 址 ! "; 
51 form.address.style.backgroundColor="#EF0000"; 
52 return false; 

53 }else{ 

54 chknew_address.innerHTML=""; 

55 form.address.style.backgroundColor="#FFFFFF"; 
56 } 

57 } 

Se 


6.6.2 动态 生成 订单 号 


从 UNIX 纪元 (格林 威 治 时间 1970 年 1 月 1 日 00:00:00) 到 当前 时 间 的 秒 数 。 其 语法 格式 如 下 : 


动态 生成 订单 号 应 用 的 是 tme0 函 数 。time0 函 数 用 于 获取 当前 的 UNIX 时 间 戳 ， 返 回 


int time (void ) 
//time0 函 数 没 有 参数 ， 返 回 值 为 UNIX 时 间 蕉 的 整数 值 


值 为 


| 这 里 就 是 以 time0 函 数 返 回 的 UNIX 时 间 戳 作为 订单 号 的 。 在 订单 处 理 页 settle_chk.php 
| 中 ,通过 time0 获 取 UNIX 时 间 截 的 整数 值 ， 并 且 将 其 赋 给 $formid 变量 作为 订单 号 。 


| 键 代码 如 下 : 
$formid=time(); // 生 成 订单 号 


6.6.3 WebBrowser 打印 


在 线 支付 模块 中 订单 的 打印 和 预览 是 非常 重要 的 两 个 功能 ， 实 现 这 两 个 功能 主要 
的 是 WebBrowser 打印 技术 。 打 印 预览 的 运行 结果 如 图 6.16 所 示 。 


强 打 印 预 览 EE 
9.，| 弓 | ~ mw 闪 1 一 “| 隐隐 人 可 | 到 | 加 
httg:1/192 te0.4.59 ~ 订单 提 立 成 功 Fa 

订单 查看 
3 9970 这 上 她: 39025132543 
下 单 人 : mr 收 贰 人 : sdisd 
5 由 ed 
eh: aas 
ER 人 和: 
订单 内 容 
EE 1 从 本 折 村 合计 
下 RT 1 3 元 3 折 2993 元 
9 元 
二 您: 订单 提交 天 = 
请 入 在 一 司 内 失信 交 吉 付 太 式 计生 世 著 月 作 的 订单 下 寺 青 导 台 知 失 让 。 
主 名 + 元 位 尝 呈 以 估 蝗 后 光 有 有 经 i3 入 且 
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图 6.16 打印 预览 
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其 关 


应 用 


WebBrowser 是 IE 内 置 的 浏览 器 控件 ， 无 需 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 | 
目标 文档 的 生成 ， 减 轻 服务 器 负荷 ， 缺 点 是 源 文档 的 分 析 操作 复杂 ， 并 且 要 对 源 文 档 中 要 | 
打印 的 内 容 进行 约束 。 
订单 打印 和 预览 主要 应 用 IE 内 置 的 WebBrowser 控件 来 实现 ， 该 控件 的 具体 参数 如 | 孜 扩 
表 6.1 所 示 。 于 


表 6.1 IE 内 置 的 WebBrowser 控件 的 参数 说 明 | Note 

参数 说 明 
document.all. WebBrowser.Execwb(7,1) 表示 打印 预览 | 
document.all. WebBrowser.Execwb(6,1 表示 打印 | 
document.all. WebBrowser.Execwb(6.6. 表示 直接 打印 | 
document.all. WebBrowser.Execwb(8,1 表示 页 面 设置 | 
document.all. WebBrowser.Execwb(]1,1) 打开 页 面 | 
document.all. WebBrowser.Execwb(2,1) 关闭 所 有 打开 的 正 窗口 
document.all. WebBrowser. Execwb(4,1 保存 网 页 
document.all. WebBrowser.Execwb(10.1 查看 页 面 属性 
document.all. WebBrowser.Execwb(17.1 全 选 
document.all. WebBrowser.Execwb(22.1 刷新 
document.all. WebBrowser.Execwb(45,1) 关闭 窗 体 无 提示 


6.6.4 支付 宝 支付 


本 模块 中 通过 支付 宝 完成 在 线 支 付 。 要 使 用 支付 宝 进 行 在 线 支 付 ， 必 须 先 在 支付 宝 中 | 
注册 一 个 账户 ， 然 后 下 载 支 付 宝 提供 的 接口 文件 ， 通 过 该 文件 实现 电子 商务 网 站 与 支付 宝 | 
的 交互 。 | 
在 支付 宝 提供 的 接口 文件 中 ， 提 供 了 完整 的 实例 程序 ， 只 要 按照 实例 程序 中 的 操作 方 | 
法 ， 将 电子 商务 网 站 中 指定 的 数据 提交 到 支付 宝 提 供 的 数组 参数 中 ,支付宝 便 会 通过 自 定 | 
义 的 方法 将 数据 提交 到 指定 的 网 站 中 。 具 体 实现 过 程 如 下 。 | 

(1) 在 forminfo.php 文件 中 ， 根 据 从 数据 库 中 读 取 的 数据 ， 定 义 支付 宝 的 数组 参数 。 | 
其 关键 代码 如 下 : | 

01 Tequire_once("alipay_service.php"): | 
02 require once("alipay_config.php"); | 
03 Sparameter = array( | 


04 "service" => "trade_create_ by_buyer"，// 交 易 类 型 , 必 填 , 实物 交易 二 trade_create_by_buyer | 
(需要 填写 物流 ) 


05 "partner" =>$partner, // 合 作 商 户 号 | 
06 "retum url"=>$returm url, // 同 步 返 回 | 
07 。 motify url" 一 Snotify url, /异步 返回 | 
08 "input_ charset" => $_input charset, /字符 集 ， 默 认为 GBK | 


09 "subject" =>$formar[0]["commo_name"]， /商品 名 称 ， 必 填 | 
10 "body" => $formar[0]["commo_ num"], /商品 描述 ， 必 填 
11 "out trade no" => $ddnumber // 商 品 外 部 交易 号 ， 订 单 号 ， 必 填 ， 每 次 测试 都 需 修 改 


“1615 


pp 内 晶 搞 半 精 角 


< 
| 12 "logistics fee"=>$yprice, // 物 流 配送 费用 
| 13 "logistics_payment"=>'BUYER _PAY'"，// 物 流 配 送 费 用 付款 方式 : BUYER_PAY( 买 家 支付 ) 
| 14 "logistics type"=>'EXPRESS', // 物 流 配 送 方式 : POST( 平 邮 )、EMS(EMS)、 
| EXPRESS( 其 他 快递 ) 

但 内 | 15 "price"=> $formar[0]["total"], // 商 品 单价 ， 必 填 

| 16 "payment type"=>"1", /默认 为 1， 不 需要 修改 

| 17 "quantity" =>"1", // 商 品 数 量 ， 必 填 

Note 18 "show_url"=> $show_url, /商品 相关 网 站 
| 19 "seller email" => $seller email // 卖 家 邮箱 ， 必 填 

200 >: 


21 Salipay = new alipay_service($parameter, $security_code,$sign type); 
22 Slink=$alipay->create_url():; 
wk] $smarty->assign("link", $link); 


(2) 在 forminfo.tpl 模板 页 中 ， 创 建 超 链接 ， 完 成 支付 操作 。 其 关键 代码 如 下 : 


01 ”<!-- 支 付 宝 支付 的 接口 操作 ， 提 交 的 数据 --> 
02 <a href="{$link}"> 支 付 宝 支付 </a> 
03 <!- -> 


说明: 
。 在 本 模块 中 ,只 是 介绍 了 PHP 中 电子 商务 网 站 与 支付 宝 进行 交互 的 方法 ， 以 及 各 种 畏 
， 助 功能 的 实现 方法 。 至 于 支付 宝 提供 的 参数 和 设置 都 是 个 人 庶 拟 的 ， 真 实 的 参数 将 在 实际 


| 的 运作 中 由 支付 宝 提供 。 在 实际 的 运作 中 ， 需 要 向 支付 宝 申 请 一 个 账户 ， 然 后 由 支付 宝 提 
| 供 相应 的 参数 ， 就 可 以 应 用 此 功能 了 。 
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( 镶 4 自学 视频 、 源 程序 : 配套 资源 mANA) 


上 网 浏览 时 ,经 常会 遇 到 这 样 的 情况 , 即 打 开 网 页 后 会 弹出 一 个 个 小 窗口 ， 
显示 一 些 广 告 、 公 告 或 者 假日 问候 等 。 在 各 大 门户 网 站 ， 这 种 现象 十 分 普遍 
尤其 是 企业 网 站 更 是 如 此 。 那 么 这 是 如 何 实 现 的 呢 ? 本 章 就 来 带领 读者 ， 亲 自 
动手 开发 一 个 假日 公告 管理 模块 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


数据 库 的 连接 与 数据 操作 
通过 超 链接 传递 值 

单 文件 上 传 

多 图 片上 传 

读 取 指 定 目录 中 的 目录 及 文件 
JQuery 控制 弹出 公告 信息 


于 于 于 于 于 至 


| 六 人 tp 重要 志 精 征 


7.1 假日 公告 管理 概述 


全/ | 7411 模块 概述 


Note | 假日 公告 管理 模块 实现 了 发 布 及 管理 假日 公告 信息 。 当 要 发 布 假日 公告 时 ， 在 “假日 
| 公告 发 布 ” 子 模块 中 上 传 公告 文件 及 图 片 即 可 ， 让 用 户 及 时 了 解 公司 的 工作 时 间 ， 并 且 可 
| 以 在 第 一 时 间 为 用 户 送 上 假日 祝福 。 假 期 过 后 ， 可 以 通过 “假日 公告 取消 ” 子 模块 ， 取 消 


发 布 的 假日 公告 ， 然 后 通过 “假日 公告 管理 ” 子 模块 将 公告 信息 删除 。 
7.1.2 系统 功能 结构 
| 假日 公告 管理 模块 的 功能 结构 如 图 7.1 所 示 。 


起 日 公告 管理 


7.1 假日 公告 管理 模块 的 功能 结构 图 


| 

| 

| 

| 

| 

| 

| [假日 公告 发 [假日 公 省 取消 [起 晶 公 告 管理 | 
| 

| < 

| 上 保 | 
| 巷 存 候 国 
| 由 | | 项 | | 全 日 全 
| 内 | | 只 | | 中 从 员 
| 人 个 = 全 
| 全 和 虽 各 
| 了 全 消 广 
| 入 息 荐 
| 

| 

| 
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| 7.1.3 程序 预览 


| 为 了 让 读者 对 假日 公告 管理 模块 有 一 个 初步 的 了 解 和 认识 ,下面 列 出 其 中 一 些 功能 模 
| 块 的 运行 效果 图 ， 供 大 家 参考 。 运 行 网 站 弹出 公告 信息 页 面 ， 效 果 如 图 7.2 所 示 。 


| 

| 

| 柚 敬 的 让 者 朋友 们 : 

| > 了 由 于 新 春 佳节 雹 日 临近 ， 本 公司 于 2011 年 1 月 31 日 到 
| 2011 年 ?月 8 日 放假， 所 以 在 此 期 间 暂 停 服务 ,请 将 您 的 问 
| 


题 发 到 技术 论坛 或 邮箱 中 ， 上 班 后 我 们 会 第 一 时 间 回 复 , 
给 您 带 来 的 不 便 ， 请 谅解 ! 


| 

| 吕 < 

| 图 7.2 运行 网 站 时 弹出 的 公告 信息 页 面 效果 
| 


。164* 


i 
图 7.3 假日 公告 管理 模块 主页 
登录 页 面 运行 效果 如 图 7.4 所 示 。 


7.4 登录 页 面 运行 效果 
假日 公告 发 布 页 面 运行 效果 如 图 7.5 所 示 。 


FE 
ET 二 


图 7.5 假日 公告 发 布 页 面 运行 效果 
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图 7.6 假日 公告 信息 取消 页 面 
假日 公告 管理 页 面 运 行 效果 如 图 7.7 所 示 。 


7.2 数据 库 设计 


| 

7.2.1 创建 数据 库 

通过 对 系统 进行 的 需求 分 析 、 业 务 流 程 设计 以 及 系统 功能 结构 的 确定 ， 规 划 出 系统 中 

| 使 用 的 数据 库 db_jiari。 数 据 库 的 结构 及 数据 表 的 情况 如 图 7.8 所 示 。 

| 胃 服务 器 : localhost ， 电 数据 库 : db_jiari 
报 作 记录 数 倒 。 类 型 吉明 

tbo 国庆 四 杀 国 义 1 WISAMN utfa_general_cj 。 假日 公告 信息 表 

tb user 厨 图 基 国 X 1 MISAM ute_general_c 管理 员 表 

2 个 表 总 计 2 MyISAM uti8_general_ci 


| 

| 

| 

| 7.8 假日 公告 管理 模块 数据 库 与 数据 表 
| 
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7.2.2 创建 数据 表 


下 面 看 一 下 各 个 表 的 结构 和 说 明 。 
回 tb 01《〔〈 假 日 公告 信息 表 ) 
假日 公告 信息 表 用 于 存储 发 布 的 假日 信息 ， 其 结构 如 图 7.9 所 示 。 
四 服务 器 : localhost ， 电 数据 库 : db_jiari > 图 表 :tb. 01 
类 型 区 理 属性 Noll 默认 麦 外 


襄 明 
这 int(4) 否 auto_increment 自动 编号 
name 。 varcharl50) utfB_unicode_ci 否 公告 主题 
perform varchar(1) 。 utfa_general_ci 否 是 否 执行 


图 7.9 假日 公告 信息 表 


回 tb_user (管理 员 信息 表 ) 
管理 员 信息 表 主 要 用 于 存储 管理 员 信 息 ， 其 结构 如 图 7.10 所 示 。 
轩 服务 器 : localhost 》 避 数据 库 : db_jiari y 加 表 :tb_user 

类 型 整理 属性 Null 默认 格外 


训 明 
uu int(4) 否 auto_increment 自动 彤 号 
name varchar(10) utf9_unicode cl 否 管理 员 名 称 
pwd 。 varchar(10) ut8_unicode ol 否 管理 员 宇 码 


图 7.10 管理 员 信息 表 
7.3 用 户 登 录 


7.3.1 ”用户 登录 概述 


用 户 登录 模块 是 假日 公告 管理 系统 的 门户 ， 只 有 登录 成 功 后 ， 才 可 以 进入 系统 主页 ， 


并 进行 假日 发 布 、 取 消 及 删除 等 操作 。 登 录 模 块 运行 效果 如 图 7.11 所 示 。 


7.11 用 户 登录 页 面 
7.3.2 ”用户 登录 功能 实现 过 程 


当 用 户 输入 登录 信息 后 ， 首 先 判断 用 户 名 和 密码 是 否 为 空 ， 然 后 再 判断 用 户 名 和 密码 | 


“I 


| 
& 


| 全 


| 
| 
是 否 正确 ， 如 果 正 确 则 登录 成 功 ， 进 入 系统 主页 ， 否 则 将 给 出 提示 信息 。 

| (1) 判断 用 户 名 和 密码 是 否 为 空 ， 应 用 的 是 checkit0 自 定义 脚本 函数 。 该 函数 存储 于 
| userphp 文件 中 ， 其 关键 代码 如 下 : 

01 <script language="javascript"> 


| 02 。 function checkitO{ // 自 定义 函数 
03 这 foml name value—""){ // 判 断 用 户 名 是 否 为 空 
| 04 alert(" 请 输入 用 户 名 !"); 
| 05 forml.name.selectO: 
| 06 Teturn false: 
| 07 
| 08 if(forml pwd.value—"){ // 判 断 密码 是 否 为 空 
| 09 alert(" 请 输入 密码 !"); 
| 10 forml.pwd.selectO: 
| 1 return false : 
| 12 } 
| 13 return tme: 
| 14 
| 15 
| 16 </scrip> 


(2) 判断 用 户 名 和 密码 是 否 正 确 ， 如 果 正 确 则 登录 成 功 ， 否 则 将 提示 用 户 名 和 密码 
不 正确 。 其 操作 是 在 user.php 文件 中 完成 的 ， 代 码 如 下 : 


| 
| 
01 <?php 
| 02 include("conn/conn.php"); // 包 含 数据 库 连 接 文件 
| 03 iflisset($_ POST['name']) and $_POST[pwd]'=nulD){ // 判 断 用 户 名 和 密码 是 否 为 空 
| 04 $select=mysql_query("select * from tb_user 
| 05 where name=".$ POST['name']." and pwd=".$_POST['pwd"]."",$conn); // 执 行 查询 操作 
| 06 if($row=mysql_num rows($select)—1){ 
| 07 $ SESSION['name']=$ POST[mame']; 
| 08 echo "<scripb>alert(' 登 录 成 功 ! ):windowlocation hreE-indexs php':</script>:"; 
| 09 jelse{ 
| 10 echo "<script>alert( 用 户 名 和 密码 不 正确 ! ):window.location href='user.php'; </script>:"; 
| 11 } 
| 2 
| I 
| 14 // 部 分 代码 省 略 
| 15 np pe ig name="imageField" onclick="return checkit0:" src="images/user 05.gif'/> 
16 // 部 分 代码 省 略 


17 <inputtype="image" name="imageField2" onclick"return false:" src="images/user 07.gif' /> 
7.4 假日 公告 发 布 


7.4.1 假日 公告 发 布 概述 


假日 公告 发 布 模块 主要 用 于 发 布 假日 公告 信息 。 进 入 假日 公告 发 布 页 面 ， 填 写 假日 名 
“68<“ 


[3 
WE: ECocumens and Senswemnet 有 


a ee 
图 7.12 假日 公告 发 布 页 面 
7.4.2 ”假日 公告 信息 发 布 实现 过 程 
(1) 创 建 form 表单 , 以 post 方式 将 数据 提交 到 promulgation_ok.php 文件 中 进行 处 理 ， 
并 且 设 置 enctype="multipart/form-data" 属 性 ， 同 时 添加 假日 名 称 、 假 日 文件 、 假 日 图 片 和 
是 否 执 行 等 表单 元 素 。 其 关键 代码 如 下 : 
01 <form action="promulgation okphp”method=-"post” enctype="multipart/form-data" name="forml" 


02 <table id=" 01" width="800" height="600" border="0" cellpadding="0" cellspacing="0"> 

03 <t> 

04 <td colspan="3"><a href="indexs php"> 

05 <img src="images/manage 01.gif' alt="" width="800" height="201" border="0" /></a></td> 
06 </tr> 

07 <t> 

08 <td><img stc="images/manage_02.gif' width="140" height="251" alt="" /></td> 
09 <td width="523" height="251" align="center" valign="top” background= 


"images/manage_03.gif'> 
10 <table width="416" height="232"> 


11 <tr> 

12 <td width="89" height="30"> 假 日 名 称 : </td> 

13 <td width="315" align="eft"><input name="name" type="text" size="40" /></td> 
14 </t> 

5 <t> 

16 <td height="30"> 假 日 文件 :</td> 

17 <td align="left"><input name="files" type="file" size="31" /></td> 
18 </> 

19 <t> 

20 <td height="30"> 假 日 图 片 1: </td> 

21 <td align="left"><?php 


"as 


49 
50 
SS 


WA 


for($a=0:$a<4:$a++){ 
> 


<input type="file" name="photo_<?php echo $a:?>" /> 
<2php 


</td> 
</tr> 
<t> 
<td height="30"> 是 否 执 行 :</td> 
<td><label> 
<input name="radio" type="radio" value="1" checked="checked" /> 
是 


<input type="radio" name="radio" value="0" /> 
否 </label></td> 
<u> 
<tr> 
<td height="30" colspan="2"><div align="center"> 
<input type="submit" name="Submit" value=" 发 布 " /> 
<input type="reset" name="Submit2" value=" 取 消 " /> 
</div></td> 
<u> 
</table></td> 
<td><img strc="images/manage 04.gif" width="137" height="251" alt="" /></td> 
</> 
<tr> 
<td colspan="3"><img src="images/manage 05.8if width="800" height="148" 


</tr> 
</table> 


</form> 


| (2) 创建 promulgation_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 完 成 假日 公告 信息 的 
| 发布 操作 。 其 关键 代码 如 下 : 


01 
02 
03 
04 
05 
06 
07 


<?php 

header("content-type:text/html:charset=utf-8"): 

include("conn/conn.php"):; 

iisset($_POST[mame']) and $_POST['Submit]—" 发 布 "){ 
Sinsert=mysql_query("insert into tb_01(name,perform) 
values(".$_POST['name'].".".$_POST['radio].")",$conn); /添加 数据 


ifisset($_FILES['files][mame])){ /判断 文件 及 图 片 是 否 存 在 

if(!is_dir("./upfile/images")){ // 判 断 指定 目录 是 否 存在 
mkdir("./upfile/images"): // 创 建 目录 

} 

Sphoto="upfile/".$_FILEST'files'][‘name']: /文件 的 存储 路 径 和 名 称 

这 move uploaded file($_FILES['files]["tmp_name'],.$photo){ /执行 上 传 
for($a=0:$a<4:$a++){ /循环 语句 

S$name=$ FILES[photo '.$a]: // 将 变量 的 名 称 保存 在 变量 中 


Spath='"upfile/images/". $name['name’]: // 定 义 上 传 文件 的 路 径 
move_uploaded file($name['tmp_name’].$path): 


“Ne 


第 7 章 假日 公告 管理 模块 


本 


18 echo "<script>alert( 上 传 成 功 ! ):windowlocation href=pronmlgation php:</scripf> | 


19 jelsef | 
20 echo "<scripPalert( 上 传 失 败 ! )windowlocation href=pronmlgation php:<scipP 

21 | | 

22 | 

23 Jelse{ | 鲜 

24 echo "<script>alert( 请 选择 上 传 文件 ! windowlocation href=prommlgation php':</script>"; | 

2 | Note | 
26 ed | 


7.5 假日 公告 取消 


7.5.1 假日 公告 取消 概述 


假期 过 后 ， 在 假日 公告 取消 页 面 中 单 击 假日 名 称 后 的 “取消 ” 超 链 接 ， 即 可 取消 该 假 
日 公告 ， 并 在 数据 库 中 删除 该 假日 信息 。 假 日 公告 取消 页 面 运行 效果 如 图 7.13 所 示 。 


和 | 旧 ] 瓜 而 | 系统 


7.5.2 ”浏览 公告 信息 


打开 假日 公告 取消 页 面 ， 默 认 将 显示 所 有 假日 信息 。 可 以 通过 select 查询 语句 查询 信 | 
息 表 中 的 数据 ， 通 过 while 语句 循环 输出 假日 公告 信息 ， 并 且 为 每 条 信息 设置 一 个 “取消 ” | 
超 链 接 ， 将 信息 的 ID 作为 参数 值 传 递 到 update.php 文件 中 。cancel.php 的 关键 代码 如 下 : 
01 ”<?php | 
02 include("conn/conn.php"): | 
03 S$select=mysql query("select * from tb 01".$conn): | 
04 while($array=mysql fetch array($select)){ | 
05 > | 
| 
| 

四 


<tr> 
07 <td height="30"><?php echo $array['name']:?> </td> 
六 


~ 可 .ip 关 表 模块 精 角 


| 
| 
| 08 dv align "center> 
| 09 <a href="update.php?abrogation=<?php echo $array['id"]:?>"> 取 消 </a> 
| 10 </div></td> 
|| al /> 
全 ff | LE <?php 
于 | | i 
14 > 


7.5.3 ”取消 假日 公告 

| 

取消 假日 公告 主要 是 通过 超 链接 参数 abrogation 传递 的 值 ， 在 update.php 页 中 执行 
| delete 语句 删除 信息 ， 取 消 假日 公告 。update.php 的 关 健 代码 如 下 : 


| 
| <Iphp 

02 header("content-type:text/html:charset=utf-8"):; 
03 include("conn/conn.php"); 

04 iflisset($ GET['abrogation']){ 


05 S$update=mysql_query("delete from tb_01 where id=".$_GET['abrogation']."",$conn); 
06 ifSupdatej{ 

07 echo "<script>alert( 假 日 信息 取消 成 功 ! ;windowJocation href'cancel.php';</script>"; 
08 }else{ 

09 echo "<script>alert( 假 日 信息 取消 失败 ! ?windowjlocation href'cancel php';</script>"; 
10 

Tn 

了 2 > 


7.6 假日 公告 管理 


| 7.6.1 假日 公告 管理 概述 


| 假日 公告 管理 模块 负责 对 上 传 的 公告 文件 及 图 片 进行 管理 , 将 过 期 的 公告 信息 文件 删 
除 ， 以 免 造 成 空间 浪费 。 假 日 公告 管理 模块 运行 效果 如 图 7.14 所 示 。 


假 | 目 [发 | 布 | 系统 


入 风光 


图 7.14 假日 公告 管理 页 面 
*。172。 


7.6.2 ”假日 公告 管理 实现 过 程 


(1) 创建 manage.php 动态 页 ， 输 出 指定 目录 下 的 文件 夹 及 文件 ， 并 实现 “上 级 目录 ” 


的 跳 转 。 其 关键 代码 如 下 : 


01 <?php 

02 if(!isset($ GET ['catalog’])||empty ($ GET [catalog])) 

03 Scurrent_directory = getcwd (): // 获 得 脚本 目录 

04 else 

05 $current directory =$_GET ['catalog]: 

06 chdir(iconv( "utf-8", "gb2312", $current directory )); /改变 当前 目录 

07 。 echo "<span class='STYLE4> 

08 ”当前 目录 :</span><span class-STYLE2>" .iconv ("gb2312", "utf-8", getcwd 0 ) . "</span><br>"; 

09 $ml= opendir (iconv ("utf-8", "gb2312", $current directory )); /打开 目录 

10 while ($gain directory=readdir ( $ml)){ /循环 读 取 目 录 中 的 目录 及 文件 

11 echo "<tr><td align='center valign=middle>": 

12 if(is dir( $gain directory )) { // 判 断 是 目录 

13 if($gain directory —".") { 

14 Scatalog = getcwd (|: // 显 示 当 前 目录 

15 Scatalog = iconv ( "gb2312", "utf-8", $catalog ): 

16 echo "<a href=manage.php?catalog=".urlencode($catalog)."><span class='stylel'> 
锁定 </span></a>"; 

17 } elseif ($gain directory —"..") { 

18 Scatalog = getcwd 0 ."\."; /上 级 目录 

Lb Scatalog = iconv ( "gb2312", "utf-8", $catalog ); 

20 if($catalog—""){ 

21 echo "<a href=manage.php?catalog=".urlencode($catalog)."><span class='stylel> 上 级 
目录 </span></a>"; 

22 jelse{ 

23 echo "<a href=manage.php?catalog=".urlencode($catalog)."><span class='stylel> 上 级 ， 
目录 </span></a>"; 

24 

25 } else { 

26 Scatalog = getcwd () . "\\$gain_directory"; // 子 目录 

27 Scatalog = iconv ( "gb2312", "utf-8", $catalog ): 

28 echo "<a href=-manage.php?catalog=".urlencode($catalog).">".iconv( "gb2312", "utf-8", 
S$gain_directory )."</a>"; 

29 } 

30 }else{ 

31 $ext = substr ( $gain directory strrpos ( $gain directory "." )): 

32 if (strtoupper ( $ext ) — ".html" || strtoupper ( $ext ) — ".gif" | strtoupper ( $ext ) 
St 

33 S$catalog = getcwd (|: 

34 Scatalog = iconv ( "gb2312". "utf-8", $catalog ): 

35 echo iconv ("gb2312", "utf-8", $gain directory ) : 

36 }else{ 

3 echo iconv ( "gb2312" "utf-8", $gain directory ): 

38 » 

39 )! 


人 


WA 


if(is dir ( $gain directory )) 
S$file size 二" 目录 ": 
else 
$file size = filesize ( $gain directory ): 
echo "<td align='center’ valign=middle>$file size</td>"; 
Screate time = date ( "Y-m-d H:i:s", filectime ( $gain_directory ) ): 
echo "<td align=center valign=middle>" . iconv ( "gb2312", "utf-8", $create time ) . "</td>": 
Supdate time = date ( "Y-m-d H:i:s", filemtime ( $gain_directory ) ): 
echo "<td align='center’ valign="middle'’>$update time</td>"; 
echo "<td align='center’ valign="middle’>"; 
if (S$gain directory —".") { 
Scatalog = getcwd 0: // 显 示 当 前 目录 
echo "删除 "; 
} elseif (Sgain directory —"..") { 
Scatalog = getcwd 0 .\."; /上 级 目录 
echo "删除 "; 
}else { 
Scatalog = getcwd 0 . "\\$gain directory":; // 子 目录 
echo "<a href='delete.php?catalog=".urlencode($catalog)." 
&filename=" .urlencode(getcwd 0) . " title=' 删 除 目录 或 者 文件 ' > 删除 </a>"; 


和 
echo “</td>"; 
echo "</tr>": 


} 
closedir ( $ml ); 
?> 


| (2) 创建 delete.php 页 ， 当 用 户 单 击 “ 删 除 ” 超 链接 时 ， 在 delete.php 页 面 接收 超 链 
| 接 传 递 的 参数 ， 完 成 删除 操作 。 其 关键 代码 如 下 : 


01 


<?php 
header("content-type:text/html:charset=utf-8"): 
$count=substr_ count($_GET['catalog]…): 
if($count>=1){ 
这 unlink($_ GET['catalog])){ 
echo "<scriptf>alert( 文 件 删除 成 功 ! ): 
window.location.href='manage.php?catalog=".urlencode($ GET['filename']).":</script>"; 
jelse{ 
echo "<script>alert( 文件 删除 失败 ! ); historybackO:</script>"; 


jelse{ 
iftis dir$_GET[catalog]){ 
if@rmdir($_GET['catalog']){ 
echo "<script>alert( 目 录 删 除 成 功 ! 小 
window.location.href='manage.php?catalog=".urlencode($ GETT'filename']).":</script>"; 
jelse{ 
echo "<scripP>alert(' 目录 删除 失败 ! "): history.backQO:</script>": 
} 


“174: 


第 7 章 假日 公告 管理 模块 


2 > 


说 明 : 
在 本 模块 中 创建 indexs.php 文件 ， 模 拟 网 站 主页 。 当 完成 假日 公告 信息 的 添加 之 后 ， 
运行 indexs php 文件 ， 将 弹出 公告 信息 ， 效 果 如 图 7.15 所 示 。 


由 于 新 春 佳节 假日 临近 ， 本 公司 于 2011 年 1 月 31 日 到 
2011 年 2 月 8 日 放假 , 所 以 在 此 期 间 暂 停 服务 , 请 将 您 的 问 
题 发 到 技术 论坛 或 邮箱 中 ， 上 班 后 我 们 会 第 一 时 间 回 复 , 
给 您 带 来 的 不 便 ， 请 谅解 ! 


| 
| 
得 散 的 读者 朋友 们 ; | 
| 
| 
| 


| 

| 

| 

| | 
图 7.15 弹出 公告 信息 效果 | 
| 


7.7 技术 提炼 


7.7.1 数据 库 的 连接 与 数据 操作 | 
1， 数据 库 的 连接 


(1) 要 操作 MySQL 数据 库 ， 首 先 必须 与 MySQL 服务 器 建立 连接 。 连接 MySQL 服 ， 
务 器 应 用 的 是 mysql_connect0 函 数 ， 其 语法 格式 如 下 。 | 
mysql_connect(hostname','username’,' 


password'): 


| 
ySQL 服务 器 的 用 户 密码 


登录 MySQL 数据 库 服务 器 的 用 户 名 


IMySQL 服务 器 的 主机 名 (或 卫 )， 如 果 省 略 端口 号 ， 默 认为 3306 


| 

该 函数 的 返回 值 用 于 表示 这 个 数据 库 连接 。 如 果 连 接 成 功 ， 则 函数 返回 一 个 资源 , 为 | 

以 后 执行 SQL 指令 做 准备 。 | 
(2) 在 连接 到 MySQL 数据 库 服 务 器 之 后 ， 接 下 来 使 用 mysql_select_db0 函 数 选择 数 

据 库 。 其 语法 格式 如 下 : | 


| 
| 
| 
| 
| 
| 
| 
\ 


rs 
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mysql_ select db ( string 数据 库 名 [resource link identifier] ) 


IMySQL 服务 器 的 连接 标识 


入 MySQL 服务 器 的 数据 库 名 称 


mysql_query("use 数据 库 名 "[resource link_identifier]); 


如 果 没 有 指定 连接 标识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 的 连接 ， 本 函数 
将 无 参数 调用 mysql_connect0 函 数 来 尝试 打开 一 个 数据 库 并 使 用 。 其 后 的 mysql_query0 


函数 调用 都 会 作用 于 活动 数据 库 。 
例如 ， 在 本 模块 中 ， 与 MySQL 数据 库 服务 器 中 的 db_jiari 数据 库 建立 连接 。 
conn\conn.php 文件 的 代码 如 下 : 
01 Sconn=mysql connect("127.0.0.1"."root"."111"); // 连 接 数据 库 服 务 器 
02 mysql select db("db jiari",$conn); // 连 接 数 据 库 
03 mysql query("set names utf8"): /设置 编码 


2. MySQL 数据 操作 


对 数据 库 中 数据 的 操作 主要 是 使 用 mysql query0 函数 ， 在 本 模块 主要 使 用 
mysql_query0 函 数 实现 添加 和 删除 信息 的 操作 。 
(1) 添加 数据 
当 用 户 发 布 假日 公告 信息 时 ， 将 向 数据 库 中 添加 一 条 假日 信息 。promulgation_ok.php 
文件 的 关键 代码 如 下 : 
01 <?php 
| 02 header("content-type:text/html:charset=utf-8"): 
| 03 include("conn/conn.php"); 
| 04 iflisset($_POST['name") and $_ POST['Submit] 一 "发 布 ){ 
| 05 /添加 数据 
| 06 。 $insert=mysql query("insert into tb_01(name.perform) values(".$_POST[mame'].".".$_ POST 
[radio].")",Sconn): 
| 


07 和 /省 略 部 分 代码 
08 Jelse{f 
| 09 echo "<script>alert( 请 选择 上 传 文件 ! ):windowJlocation href=prommlgation php':</script>"; 
| ‘ee 
| 1 > 
| (2) 删除 数据 
| 当 用 户 取消 假日 公告 时 ， 将 删除 数据 库 中 的 假日 信息 。update.php 文件 的 代码 如 下 : 
01 <?php 


| 
| 02 header("content-type:text/html:charset=utf-8"): 
| 03 include("conn/conn.php"): 

| 04 iflisset($ GET['abrogation"])){ 
| 
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05 $update=mysql query("delete from tb 01 where id=".$_ GET['abrogation']."".$conn);: + | -- Formatted: Line spacing 
06 if($update){ least 12.5 pt 
07 echo "<script>alert( 假 日 信息 取消 成 功 ! ):windowlocation href='cancel php':</script>"; 
08 jelse{ 
09 echo "<scripPalert( 假日 信息 取消 失败 ! :windowJlocation href='cancel php':</script>"; 
10 
1 } 
120 "> 


7.7.2 ”通过 超 链接 传递 值 
通过 超 链接 传递 值 主要 使 用 的 是 <a> 标 记 ， 其 语法 格式 如 下 。 


| 
| 
| 
| 
| 
<a href=URL name=name target=target> 链 接 文 字 </a> + 上 | Formatted: Line spacing 
| least 14 pt 


a 标记 的 属性 介绍 如 下 。 
回 href 属性 : 指定 所 链接 文件 的 URL 路 径 。 这 个 路 径 可 以 是 相对 路 径 ， 也 可 以 是 绝 
对 路 径 。 | 
回 name 属性 :指定 页 面 的 锚 点 名 称 。 如 果 需 要 链接 到 对 应 的 锚 点 位 置 ， 需 要 在 锚 
点 名 称 前 添加 一 个 “#” 字 符 。 
回 target 属性 : 指定 要 打开 的 链接 所 使 用 的 浏览 器 窗口 名 称 。 可 以 使 用 自 定 义 的 窗 | 
口 名 称 。 也 可 以 使 用 下 面 4 个 内 置 的 窗口 名 称 ， 其 中 前 两 个 比较 常用 。 | 
4 _self: 在 当前 窗口 中 打开 链接 文件 ， 此 为 默认 值 。 | 
4 _blank: 开启 一 个 新 的 窗口 打开 链接 文件 。 
今 “_parent: 在 父 级 窗口 中 打开 文件 ， 常 用 于 框架 页 面 。 | 
人 _top: 在 顶层 窗口 中 打开 文件 ， 常 用 于 框架 页 面 。 | 
本 模块 中 主要 在 取消 公告 及 删除 公告 文件 中 使 用 了 超 链接 。 
(1) 假日 公告 取消 页 面 中 的 “取消 ” 超 链接 : | 


<a href="update.php?abrogation=<?php echo $array["id]:?>"> 取 消 </a> | 有 eb yp Line spacin8 
east 14 pt 


(2) 假日 公告 管理 页 面 中 的 “删除 ” 超 链 接 : 


<a href='delete php?catalog=".urlencode($catalog)."&filename=" . urlencode(getewd 0) .mtitle= 删 除 <- -- | 
目录 或 者 文件 '> 删 除 </a> least 14 pt 


说 明 : 
获取 超 链 接 参 数 传递 的 值 ， 应 用 的 是 5_GET[] 全 局 变量 。 


7.7.3 单 文件 上 传 | 
| 


本 模块 中 上 传 的 单 文 件 是 公告 信息 文件 ， 主 要 使 用 了 mkdir0 函数 和 | 
Immove uploaded file0 函 数 。 下 面 分 别 对 其 进行 介绍 。 


.177 . 


| 
| >] 


(1) mkdir0 函 数 
判断 某 文 件 是 否 存在 , 并 且 是 否 可 写 , 如 果 满足 上 述 条 件 则 返回 True, 否则 返回 False。 
| 其 语法 格式 如 下 : 


| 
| 
| 
| 
| 
bool mkdir(string pathname [. int mode]) 


NOLE 其 中 参数 说 明 如 表 7.1 所 示 。 
表 7.1 mkdir() 函 数 的 参数 说 明 


| 
| 参数 说 有 明 人 i| Formatted: 表 栏 名 
pathname 必要 参数 。 用 于 指定 新 建 目录 的 名 称 

| mode 可 选 参数 。 用 于 指定 新 建 目录 的 模式 


| (2) move_uploaded_file0 函 数 

| 该 函数 应 用 POST 方法 实现 文件 的 上 传 。 其 语法 格式 如 下 : 
| bool move_uploaded file(string filename. string destination) 
| 
| 


其 中 参数 说 明 如 表 7.2 所 示 。 
表 7.2 move_uploaded_file() 函 数 的 参数 说 明 


必要 参数 。 指 定 要 上 传 的 文件 地 址 
destination 必要 参数 。 上 传 到 服务 器 后 的 存储 目录 及 名 称 


| 
| 
| 
| 
| 
| 应 用 POST 方法 上 传 文件 时 , 必须 在 上 传 表单 的 <form> 标 记 中 添加 enctype="multipart/ 
| form-data" 属 性 。 

| 


| 7.7.4 多 图 片上 伟 


| 本 模块 在 上 传 公告 图 片 时 ， 采 用 了 多 图 片上 传 方式 ， 使 用 for 循环 语句 循环 获取 图 片 
信息 ， 然 后 使 用 move_upload_file0 函 数 执 行 上 传 操作 。 


for 语句 是 PHP 中 最 复杂 的 循环 控制 语句 ， 拥 有 3 个 条 件 表达 式 。 其 语法 格式 如 下 : 
for (exprl: expr2; expr3){ 
statement 
} 
其 中 参数 说 明 如 表 7.3 所 示 。 / 
表 7.3 for 循环 语句 中 的 参数 介绍 了 


Formatted: Line spacing 
least 13 pt 


Formatted: Line spacing 
least 13 pt 


Formatted: Line spacing 
least 13 pt 


必要 参数 。 第 一 个 条 件 表达 式 ， 在 第 一 次 循环 开始 时 被 执行 
必要 参数 。 第 二 个 条 件 表达 式 ， 在 每 次 循环 开始 时 被 执行 ， 决 定 循环 是 否 继续 * ，… 


4 


Formatted: Line spacing 
least 13 pt 


expr3 必要 参数 。 第 三 个 条 件 表达 式 ， 在 每 次 循环 结束 时 被 执行 A 
statement 必要 参数 。 满 足 条 件 后 ， 循 环 执行 的 语句 二 | least13pt 


j »。178°* 
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其 执行 的 过 程 : 首先 执行 表达 式 1; 然后 执行 表达 式 2， 并 对 表达 式 2 的 值 进行 判断 ，| 


如 果 值 为 真 ， 则 执行 for 循环 语句 中 指定 的 内 嵌 语 句 ， 如 果 值 为 候 ， 则 结束 循环 ， 跳 出 for | 


循环 语句 ， 最 后 执行 表达 式 3 (切记 是 在 表达 式 2 的 值 为 真 时 )， 返 回 表 达 式 2 继续 循环 执 
行 。for 循环 语句 的 操作 流程 如 图 7.16 所 示 。 


图 7.16 for 循环 语句 的 流程 图 
7.7.5 读 取 指 定 目录 中 的 目录 及 文件 


读 取 指定 目录 中 的 目录 及 文件 主要 使 用 opendir0 函 数 、getewd0 函 数 、iconv0 函 数 以 
及 while 循环 语句 ， 循 环 输出 目录 文件 。 
(1) opendir() 函 数 
打开 指定 的 目录 ， 并 返回 该 目录 的 标识 。 其 语法 格式 如 下 : 
resource opendir(string path) 
参数 path 为 该 函数 所 要 打开 目录 的 路 径 。 
(2) getcwd0 函 数 
返回 当前 的 工作 目录 。 其 语法 格式 如 下 : 
string getcwd(void) 
该 函数 的 参数 为 空 。 
(3) iconv0 函 数 
将 一 种 已 知 的 字符 编码 格式 A 转换 成 另外 一 种 编码 格式 B。 其 语法 格式 如 下 : 
String iconv ( string in_charset. string out_charset, string str ) 
参数 说 明 如 下 。 
in_charset: 一 种 已 知 将 被 转换 的 编码 格式 A。 
out_charset: 另 一 种 将 要 转换 成 的 编码 格式 B。 
str: 指定 被 转换 的 字符 串 。 该 函数 转换 成 功 返 回 字 符 串 ， 否 则 返回 false。 
(4) while 循环 语句 
while 循环 语句 的 作用 是 反复 地 执行 某 一 项 操作 ， 是 循环 控制 语句 中 最 简单 的 一 种 ， 


“79. 


人 


也 是 最 常用 的 一 种 。while 循环 语句 对 表达 式 的 值 进 行 判 断 ， 当 表达 式 的 值 为 非 零 时 ， 执 
| 行 while 语句 中 的 内 嵌 语 句 ， 当 表达 式 的 值 为 0 时， 则 不 执行 while 语句 中 的 内 嵌 语 句 。 
| 该 语句 的 特点 是 : 先 判断 表达 式 ， 后 执行 语句 。while 循环 控制 语句 的 操作 流程 如 图 7.17 


J | | 所 示 。 
| | 


图 7.17 while 循环 控制 语句 的 操作 流程 


其 语法 格式 如 下 : 
while (expD){ 产 
statement: 先 判断 条 件 ， 当 条 件 满足 时 执行 语句 块 ,否则 
不 向 下 执行 
} " 


只 要 while 表达 式 expr 的 值 为 TRUE, 就 重复 执行 嵌 套 中 的 statement 语句 ; 如 果 while 
表达 式 的 值 一 开始 就 是 FALSE， 则 循环 语句 一 次 也 不 执行 。 


7.7.6 jQuery 控制 弹出 公告 信息 


在 假日 公告 管理 系统 中 , 控制 公告 信息 的 输出 应 用 的 是 jQuery 技术 。 下 面 对 具 体 应 用 
的 方法 进行 介绍 。 
(1) $0: 用 于 替代 document.getElementById0。 其 语法 格式 如 下 : 
Var someElement = $("#myId"): 
例如 : 通过 jQuery 获取 DOM 对 象 中 的 元 素 ， 代 码 如 下 : 
01 S$("div m"): 
02 S$("div.admin"):; 
03 S$("div#mysqlid "): 
04 S$("table a".content); 
第 一 行 代码 获取 所 有 标签 下 的 m 元 素 ; 第 二 行 代码 获取 class 为 admin 的 元 素 ; 第 三 
行 代 码 获 取 标 签 下 面 id 为 mysqlid 的 元 素 ; 第 四 行 代码 获取 content 为 上 下 文 的 table 中 所 
有 的 连接 元 素 。 
(2) slideToggle(speed,[callback]): 通过 高 度 变化 来 切换 所 有 匹配 元 素 的 可 见 性 ， 并 在 
切换 完成 后 可 选 地 触发 一 个 回调 函数 。 
这 个 动画 效果 只 调整 元 素 的 高 度 ， 可 以 使 匹配 的 元 素 以 “滑动 ”的 方式 隐藏 或 显示 。 
参数 说 明 如 下 。 


“Ne 
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回 ”speed (String,Number): 3 种 预定 速度 之 一 的 字符 串 ("slow", "normal", or "fast") 
或 表示 动画 时 长 的 毫秒 数值 (如 1000) 

callback (Function): 可 选 参数 ， 在 动画 完成 时 执行 的 函数 。 

例如 ， 用 600 毫秒 缓慢 地 将 段落 滑 上 或 滑 下 。 


me 
OLe 


又 如 ， 用 200 毫秒 快速 将 段落 滑 上 或 滑 下 ， 之 后 弹出 一 个 对 话 框 。 


$("p").slideToggle("fast",functionO{ 
alert("Animation Done."): 
»; 
(3) ready(fo): 当 DOM 载 入 就 绪 ， 可 以 查询 及 操纵 时 ， 绑 定 一 个 要 执行 的 函数 。 这 | 
个 方法 是 window.load 事件 的 替代 方法 。 其 语法 格式 如 下 : | 


ready(fh) 


| 

参数 血 (Function) 为 在 DOM 就 绪 时 执行 的 函数 。 可 以 为 这 个 参数 任意 起 一 个 名 字 ， 

并 因此 可 以 不 再 担心 命名 冲突 而 放心 地 使 用 $ 别 名 。 | 
| 


说 明 : | 

在 使 用 ready(fn) 时 ， 必 须 确保 在 <body> 元 素 的 onload 事件 中 没有 注册 函数 ,否则 不 会 
触发 $(document) ready0 事 件 。 可 以 在 同一 个 页 面 中 无 限 次 地 使 用 $(document) ready0 事 件 。 | 
其 中 注册 的 函数 会 按照 代码 中 的 先后 顺序 依次 执行 。 | 


例如 ， 通 过 $(document).ready(fn) 代 替 body 标签 的 onload 事件 ， 代 码 如 下 : 
01 S$(document).ready(functionO{ 

02 alert("jQuery"): | 

3 | 

使 用 $(document).ready0 的 简写 ， 同 时 内 部 的 jQuery 代码 依然 使 用 $ 作 为 别名 ， 而 不 必 | 

管 全 局 的 $ 是 什么 。 其 jQuery 代码 如 下 : | 

| 

| 

| 


jQuery(function($) { 
/可 以 在 这 里 继续 使 用 $ 作 为 别名 … 
D; 
上 述 $(document).ready0O 的 简写 替换 的 是 如 下 代码 : | 
<body onload="alert(jQuery): > 
(4) css(name): 访问 第 一 个 匹配 元 素 的 样式 属性 。 返 回 值 为 string， 参 数 name 为 要 
访问 的 属性 名 称 。 | 
例如 ， 取 得 第 一 个 段落 的 color 样式 属性 的 值 。 
SC"p").css("color"): | 
(5) css(name,value): 在 所 有 匹配 的 元 素 中 ， 设 置 一 个 样式 属性 的 值 。 数 字 将 自动 转 


| 
.181 。 WD 
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其 中 ， 参 数 name 为 属性 名 ， 参 数 value (String, Numben) 是 属性 值 。 
| 例如 ， 初 始 状态 下 隐藏 下 拉 列 表 。 
$(#cs").css("display""none"): ”// 初 始 状态 下 使 城市 下 拉 列 表 不 可 见 


同时 还 要 创建 div 标签 ， 输 出 公告 信息 。 在 输出 公告 信息 之 前 ， 要 读 取 数 据 库 中 的 数 
据 ， 判 断 是 否 输出 公告 信息 。 如 果 指 定 字段 值 为 1， 则 包含 上 传 的 html 文件 ， 即 输出 公告 
信息 ; 否则 不 输出 div 标签 中 的 内 容 。 其 关键 代码 如 下 : 


在 假日 公告 管理 系统 中 ， 需 要 在 输出 公告 信息 的 网 页 中 载 入 jQuery 库 文件 ， 编 写 
JavaScript 脚本 ， 控 制 公告 信息 的 输出 。 其 关键 代码 如 下 : 
| 01 <aript sre"js/iqueryjs"></script> 
| 02 <script> 
| 03 function showLayerNoticeO{ 
| 04 S$("#Layer110").slideToggle(1000); 
| 05 } 
| 06 S$(document) .ready(functionO{ 
| 07 $("#Layer110").css("left", (screen.wWidth-520)/2): 
| 08 setTimeout("showLayerNotice0" 1300) : 
| 09 》: 
| 10 </scrip> 
| 
| 


| 

| Ol <Iphp 

| 02 include("conn/conn.php"); // 连 接 数 据 库 

| 03 Sresult=mysql query("selectperform from tb 01".$conn); /执行 查询 语句 

| 04 $myrow=mysql fetch array($result): // 获 取 查 询 结 果 

| 05 ”过 Smyrow['perform'] 一 1){// 判 断 指定 字段 的 值 ， 如 果 等 于 1 则 输出 公告 信息 ， 否 则 不 输出 
| 06 > 


07 <div id="Layerl10" 
08 style="position: absolute: border:3px solid #CCCCCC: left: 50px: top: 50px: z-index: 1; font-size: 


| 0px: display: none'"> 
| 09 <?php 
| 10 include("upfile/index.html"): // 包 含 公告 文件 
| 11 2 
| 12 </div> 
| 13 <?php 
| 14 } 
3 
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投票 系统 模块 
( 名 + 自学 视频 、 源 程序 ， 配套 资源 mA8\) 


针对 基 一 主题 ， 在 进行 网 上 调查 时 ， 投 票 系统 必 不 可 少 。 通 过 该 系统 可 以 
较为 客观 、 真 实地 收集 、 分 析 广 大 用 户 对 该 主题 的 意见 及 想法 ， 以 便 作 出 正确 
的 判断 ， 从 而 有 针对 性 地 加 以 改进 或 完善 ， 为 用 户 提供 更 好 的 服务 。 在 本 章 中 
将 向 广大 读者 介绍 一 些 投 票 的 实现 方法 和 分 析 投票 结果 的 方法 ， 以 及 在 投票 中 
如 何 限制 重复 投票 等 技术 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


了 单 选 投票 与 多 选 投票 的 关系 

WI 通过 柱 形 图 与 饼 图 分 析 投票 结果 
I 通过 IP 限制 重复 投票 

WI 利用 Cookie 变量 限制 重复 投票 
PI 动态 生成 投票 主题 及 选项 


a tp 六 量 搞 志 持 角 


8.1 投票 系统 概述 


8.1.1 ”模块 概述 


| 投票 系统 的 主要 作用 是 根据 一 个 主题 ， 以 提供 的 投票 选项 为 条 件 ， 选 出 正确 、 合 理 的 
| 一 项 或 者 多 项 内 容 ， 从 而 更 好 地 了 解 广大 用 户 对 该 主题 的 意见 及 想法 。 为 了 使 其 功能 更 加 
| 完善、 合理， 本 系统 还 开发 了 后 台 管 理 功能 ， 对 投票 和 网 站 进行 管理 。 


”8.1.2 ”功能 结构 


票 系统 的 功能 结构 如 图 8.1 所 示 。 


漳 哗 离 册 酒水 
滑 吨 懂 了 圣 酒 选 


语 莹 次 册 油水 
半 席 芯 吾 油水 
Es 


图 8.1 投票 系统 的 功能 结构 图 


8.1.3 ”程序 预览 


| 投票 系统 是 由 多 个 功能 模块 组 成 的 ， 由 于 篇 幅 有 限 ， 下 面 仅 列 出 几 个 典型 功能 模块 的 
| 运行 效果 图 ， 让 读者 对 本 系统 有 一 个 初步 的 了 解 和 认识 。 
| 票 主题 浏览 页 面 运行 效果 如 图 8.2 所 示 ， 其 主要 功能 是 给 浏览 者 提供 一 个 投票 的 


| 
| 
| 平台 。 
| 
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图 8.2 投票 主题 浏览 页 面 运行 效果 | 
通过 柱 形 图 分 析 投 票 结果 ， 其 运行 效果 如 图 8.3 所 示 。 


Stat Vote Result 


1400 

ee 
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~ es Ey 
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400 六 
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陆 一 乔 伊 
Stat Vote Result 
图 8.3 ”通过 柱 形 图 分 析 投票 结果 
通过 3D 饼 图 分 析 投 票 结果 ， 其 运行 效果 如 图 8.4 所 示 。 


Stat Vote Result 


33% 


0% 


图 8.4 通过 3D 饼 图 分 析 投 票 结果 


后 台 管 理 首页 的 运行 效果 如 图 8.5 所 示 ， 其 主要 功能 是 对 投票 的 主题 内 容 和 投票 进行 | 
管理 。 
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售 收 记 本 站 
设 为 首页 


对 | i 
管理 技 票 内 五 全 秀 员工 评选 公司 和 内 部 的 优秀 员工 。 
dat 评选 公 i 人 
Not e 5 和 于 村 优秀 负责 人 评选 公司 内 部 优秀 负责 人 
到 我 最 豆 允 的 人 Et 入 
re 
FF 部 分 最 好 4 瑟 书 请 选择 出 FE 部 门 暴 好 的 加 书 
证 是 FHP 最 优秀 5 员工 请 达 择 认 是 BE 最 优秀 的 员工 ， 
投票 内 容 


图 8.5 后 台 管 理 首页 运行 效果 
管理 投票 主题 页 面 的 运行 效果 如 图 8.6 所 示 , 其 主要 功能 是 给 浏览 者 提供 投票 的 主题 ， 
以 及 对 已 有 投票 主题 的 删除 。 


岂 、 当前 位 置 管 理 投票 主题 


主题 : 公司 春季 旅游 地 点 选择 


为 了 缓解 员工 工作 的 压力 ， 现 组 织 一 次 春季 旅游 ， 时 间 为 3 
j， 天 。 提供 多 个 旅游 景点 请 大 家 选择 * 


[本 王 ] 

| 

优秀 员工 

优秀 负责 人 

我 最 喜欢 的 人 

PIP 部 分 最 好 的 图 书 

谁 是 PHP 最 优秀 的 员工 

优秀 系统 评选 


图 8.6 管理 投票 主题 页 面 运 行 效果 


8.2 数据库 设计 


”8.2.1 数据 库 设计 

| 

投票 系统 模块 使 用 的 数据 库 是 db_vote， 其 中 包含 6 个 数据 表 ， 分 别 是 tb_voter (用户 
| 信息 表 )、tb_vote_more (多 选 投 票 表 )、tb_vote_ more ip (参与 多 选 投票 者 的 ip 表 )、 
”tb_vote_odd ( 单 选 投票 表 )、tb_vote_odd ip (参与 单 选 投票 者 的 记 表 )、tb_vote_subject ( 投 
| 票 主题 表 )。db_vote 数据 库 结构 如 图 8.7 所 示 。 
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胃 服务 器 : localhost ， 时 数据 库 : db_vote 

表 雪 作 记录 数量 。 类 型 茎 理 讨 
也 b_voter 上 雷 加 比 便 XX 5 MySAM ”9b2312_chinese_cl 。 存 结 注 册 的 用 户 信息 
th_vote_more 下 加 线 国 XxX 7 MYSAM 。 gb2312_chinese_ci 。 存 鱼 多 选 投票 的 内 宫 
也 vote_more 种 河 办 吾 钴 恒基 0 IISAM 。 9b2312_chinese_cl 通过 rp 控制 多 选 投票 中 的 重复 投票 
了 b vote _odd 加 本 加 线 硬 XX 13 MYSAM 。 gb2312_chinese_cl 存 佬 单 选 投票 内 容 ! 和 
vow_odd ip 上 加 半 国 xXx 5 MSAM 。 gb2312_chinese_c 。 裔 过 ?控制 单 选 投票 中 的 重复 投票 | 食 作 | 
th_vote_subject 军国 革 硬 XX 5 MyISAM 。 gb2312_chinese_ci 。 存 铺 添加 的 投票 主 是 | 二 [一 


6 个 表 总 计 4 MYISAM gb2312_chinese_ci | 
图 8.7 投票 系统 数据 库 结构 


8.2.2 ”数据 表 设计 


下 面 对 投 票 系统 使 用 的 db_vote 数据 库 中 的 数据 表 结 构 进 行 分 析 。 由 于 篇 幅 有 限 ， 在 
此 仅 以 tb_vote_ more 和 tb_vote_subjet 为 例 进行 介绍 。 | 
tb_vote_more( 多 选 投 票 表 ) 的 结构 如 图 8.8 所 示 。 


四 服务 器 : localhost ， 忆 数据 库 : db_vote ， 国 表 : tb_vote_more 

字段 类 型 整理 属性 Null 默认 要 外 襄 明 
也 vote id int(10) 否 auto_increment 自动 编号 
th_vote_subject varchar(80) gb2312_chinese_ci 否 投票 主题 
th_vote_content mediumtexd gb2312_chinese_d 理 投票 内 容 
th_vote_counts varchar(50) 9b2312_chinese_ol 否 投票 数 
th_vote type varchar(50) gb2312_chinese_cl 否 投票 方式 
‘th_vote_explain 。 mediumtexd gb2312_chinese_ol 否 投票 说 明 
th_vote photo varchar(80) 9b2312_chinese_ 否 投票 内 容 多 图 片 


图 8.8 多 选 投票 表 | 
tb_vote_subject (投票 主题 表 ) 的 结构 如 图 8.9 所 示 。 | 


图 服务 器 : localhost ， 局 数 掘 库 : db_vote 》 团 表 : tb_vote_subject | 

字段 类 型 整理 属性 ”Null 默认 本 外 讽 明 | 

也 vote sublect id Intf1o) E3 auto_increment ”投票 主 是 89ID | 
tb_vote_subject_name varchar(80) gb2312_chinese_ci 否 投票 主题 名 称 
tb_vote_subject_text text 9b2312_chinese_ci 否 投票 主题 指示 


图 8.9 投票 主题 表 
8.2.3 连接 数据 库 


由 于 本 模块 大 部 分 页 面 都 需要 使 用 数据 库 ， 如 果 每 页 都 编写 相同 的 数据 库 连 接 代码 ， | 
会 显得 十 分 繁 天 ， 所 以 在 此 将 数据 库 连 接 代码 单独 存 入 一 个 PHP 文件 conn.php 中 ， 并 将 | 
该 文件 存储 于 conn 文件 夹 中 ， 在 需要 与 数据 库 连接 的 页 面 中 ， 使 用 包含 语句 调用 该 文件 | 
即 可 。 该 模块 实现 与 数据 库 连接 的 代码 如 下 : 


Oo Zipp | 
02 $conn=mysql connect("localhost","root","111") or die(' 连 接 失 败 :'. mysql error(0); // 连接 | 


服务 器 | 
03 ”mysql_select_db("db_vote",$conn) or die(' 数 据 库 选择 失败 :' . mysql_error0); /选择 数据 库 | 
04 mysql query("set names gb2312"): /设置 数据 库 编 码 格式 
DS 
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8.3 投 票 


| 在 本 模块 中 ， 浏 览 者 想 要 对 某 一 主题 进行 投票 ， 必 须 先进 行 注册 。 注 册 成 功 后 ， 以 指 


定 的 用 户 身份 登录 系统 ， 然 后 才 可 以 进行 投票 。 在 本 节 中 将 对 用 户 的 注册 、 登 录 、 投 票 主 
题 浏览 和 投票 进行 详细 讲解 。 


8.3.1 用 户 注册 


| 用 户 注 册 的 功能 是 存储 投票 者 的 个 人 信息 ,同样 也 是 为 投票 者 提供 一 个 登录 本 系统 的 
”钥匙 。 用 户 注册 模块 的 运行 效果 如 图 8.10 所 示 。 


在 线 投票 系统 


ZAIXIANTOURIAOXITONG 


会 员 名 : Parsoft 
定 码 :; [eee 


邮箱 地 址 ，|sp@nrbccd. con (请 填写 正确 的 E-mail 地 址 ! ) 
电话 号 码 ，[1360433**** 
联系 地 址 [长 春 市 


提交 注册 信息 重新 填写 注册 信息 


技术 服务 热线 : 0431-64978981 84978982 
Copyright@ mw ningrisoft. com 类 1 Rights Reserveal 


图 8.10 用 户 注册 页 面 


| 首先 ,创建 一 个 register.php 文件 ,在 该 文件 中 添加 用 户 的 个 人 信息 ,并 且 应 用 JavaScript 
”脚本 对 添加 的 信息 进行 判断 ， 将 数据 提交 到 register_ok.php 文件 中 。 其 中 应 用 的 JavaScript 
”脚本 语句 如 下 : 

| 01 <script> 


02 function check email(tb_voter maiD{ 
03 Var str=tb_voter_mail; 


| 04 Var Expression=Aw+([-+.]w+)*@\Ww+([-.J 人 w+)*ANw+([-.] 人 w+)*/; 。// 判 断 邮 箱 地 址 的 
| 格式 是 否 正确 
| 05 var objExp=new RegExp(Expression); 
06 if(objExp.test(str)==true){ 
| 07 Teturn true; 
| 08 Jelse{ 
| 09 return false: 
10 } 
i 
12 </script> 


3 <script language="JavaScript" type="text/javascript"> 


了 。188 。 


14 function check input(form){ 


15 if(form.tb_voter name.value—""){ 
16 alert(" 请 填写 会 员 名 称 ! "); 
17 form.tb_voter name.select(); 
18 return(false); 

19 hn 

20 if(form.tb_voter pass.value—""){ 
2 alert(" 请 输入 密码 !"); 

22 form .tb_voter pass.focus(); 
23 Tetum(false); 

24 

25 if(form.tb_voter mail.value—""){ 
26 alert(" 请 输入 E-mail 地 址 !1"); 
2 form.tb_voter mail.select(); 

28 retum(false); 

29 由 

30 if(!check_email(form.tb_voter_mail.value){ 
31 alert(" 您 输入 的 E-mail 地 址 的 格式 不 正确 !"); 
3 form.tb_voter mailfocusO:; 

33 Tetum(false): 

34 } 

35 if(form.tb_voter tel.value—""){ 

36 alert(" 请 输入 电话 !"); 

37 form.tb_voter tel.select|:; 

38 return(false): 

39 } 

40 if(form.tb_voter_address.value—""){ 
41 alert(" 请 填写 地 址 ! "); 

42 form.tb_voter_address.selectO; 
43 return(false): 

a Hh 

45 

46 return(true); 

47 

48 </script> 


然后 ， 创 建 register_ok.php 文件 ， 将 表单 提交 的 数据 存储 到 指定 的 数据 表 中 。 其 代码 


01 <?php 
02 session_ start(); 
03 include_once("conn/conn.php"); 


04 if(isset($_ POST['submit"])){ // 处 理 注册 用 户 提交 的 数据 
05 S$tb_voter name=trim($ POST['tb_voter name']): 

06 S$tb_voter pass=md5($ POST['tb_voter pass"]); 

07 S$tb_voter mail=trim($ POST['tb_voter mail"]); 

08 S$tb_voter tel=trim($ POST[tb_voter tel]): 

09 S$tb_voter address=trim($ POST['tb_ voter address]): 

10 S$tb_voter ip=getenv("REMOTE ADDR"); 1/ 获取 投票 者 的 人 Pp 


11 $query=mysql query("insert into tb voter(tb voter name.tb voter pass.tb voter mail.tb voter teltb voter 
“189's 


到 VAI 


address,tb_voter ip) 
| 12 values('$tb_voter name','$tb_voter pass','$tb_voter mail','$tb_voter tel','$tb_voter address', 
| '$tb_voter_ip)",$conn); 
| 这 Squery){ 

$_ SESSION['tb voter _ name']=$tb_voter name; ”// 通 过 seesion 变量 存储 投票 


$_SESSION['tb_voter mail]=$tb_voter mail: // 通 过 seesion 变量 存储 投票 
echo "<script>alert(' 注 册 成 功 !");window.location.href='main.php';</script>"; 


jelse{ 
echo "<script language='javascript>alert(' 对 不 起 ,注册 失败 !):historybackO; 


| 
| 20 } 

| MI 
8.3.2 用 户 登录 


用 户 登 录 模块 主要 是 供 投 票 系统 的 注册 用 户 进 行 登录 操作 。 该 页 面 主 要 由 两 个 文件 组 
| 成 ， 一 个 是 用 户 登 录 文 件 ， 包 括 用 户 名 、 密 码 ; 另 一 个 是 用 户 登录 的 处 理 文件 。 用 户 登 录 
页面 如 图 8.11 所 示 。 
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| 用 户 名 : nr 
密码 : eeeeee 


今天 是 : 2011-05-10 [Esa 


技术 服务 热线 : 0431-84978981 84978982 
Copyright 回 www.mingrisoft com 各 1 Rights Reserved! 


图 8.11 用 户 登录 页 面 


| 在 用 户 登 录 文件 中 ,创建 一 个 表单 ， 其 中 包括 用 户 名 和 密码 两 个 文本 框 ,将 表单 中 的 内 
| 容 提 交 到 enter_ok.php 文件 中 。 在 index.php 文件 中 ， 创 建 表单 和 表单 元 素 的 关键 代码 如 下 : 


| 01 <form action="enter ok.php" method="post" name="forml1" id="forml"” onSubmit="return 
| check user(this)"> 


| 02 <tr> 

| 03 <td width="87" align="center"> 用 户 名 : </td> 

| 04 <td width="139"><input type="text" name="tb_user" size="18" /></td> 

| 05 <td width="69" align="center"> 密 码 : </td> 

| 06 <td width="148"><input type="password" name="tb_pass" size="18" /></td> 
| 07 <td width="63"><input type="submit" name="Submit' value=" 登 录 "></td> 

| 08 <td width="186"> 今 天 是 : <?php echo date("Y-m-d"):?></td> 

| 09 <td width="58">&nbsp:</td> 
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10 </tr> 
11 </form> 


在 enter_ok.php 文件 中 ， 实 现 对 表单 提交 的 数据 进行 处 理 。 主 要 是 判断 用 户 提交 的 用 | 
户 名 和 密码 是 否 正确 ， 如 果 正 确 则 可 以 进行 登录 ， 否 则 将 不 能 登录 。 代 码 如 下 : 


01 <?php 

02 session start(); 

03 include("conn/conn.php"); // 连 接 数 据 库 

04 S$tb user=$ POST[tb_user]:; // 获 取 提 交 用 户 名 


05 ”Stb_pass=md5($_ POST['tb pass]): ”// 获 取 提 交 密 码 
06 ifisset($_ POST['tb_user]) && isset($_ POST['tb pass])){ 


07 $query=mysql query("select * from tb voter where tb_ voter name='$tb user and | 
tb_voter pass='$tb_pass'"); | 

08 if(Squery){ | 

09 $_SESSION["tb_user"]=$tb_user; | 

10 echo "<script>alert(' 登 录 成 功 ! ");window.location.href='main.php';</script>"; | 

11 jelsef | 

12 echo "<script>alert(' 您 输入 的 用 户 名 或 密码 不 正确 ! ");window.location.hre 仁 
‘index.php';</script>"; 

13 } 

14 }else{ 

15 echo "<script>alert(' 请 您 正确 登录 , 谢谢 !");window.location.href="index.php';</script>" 

16 } 

17 ?> 


8.3.3 ”投票 主题 浏览 


在 投票 主题 浏览 页 面 中 ， 将 输出 本 系统 提供 的 投票 主题 ， 同 时 还 将 显示 一 些 超 链 接 ， 
于 进入 相应 的 投票 内 容 提交 页 面 中 。 投 票 主题 浏览 页 面 的 运行 效果 如 图 8.12 所 示 。 
7 一 


aeaAwToupwwonirone 


py 


crerrie@, oe 


图 8.12 ”投票 主题 浏览 页 面 | 


投票 主题 浏览 是 通过 main php 文件 来 完成 。 首先 , 判断 用 户 是 否 正确 登录 , 如果 | 
正确 登录 ， 则 可 以 进入 本 系统 中 ， 和 否则 将 弹出 提示 信息 “请 您 正确 登录 1”， 返 回 到 用 | 
户 登录 页 面 。 | 
然后 ， 应 用 SELECT 查询 语句 ， 从 投票 主题 表 中 读 取 出 所 有 投票 主题 的 数据 ， 将 数据 | 
进行 循环 输出 ， 并 且 设 置 超 链接 ， 链 接 到 投票 内 容 提 交 页 面 。main php 的 关键 代码 如 下 : 


和 


01 < ?php session start(); 
02 include("conn/conn.php"); 
03 ifisset($_ SESSION['tb_ user])) { 


04 ?> 

05 a 

06 <?php 

07 $result=mysql_query("select * from tb_vote_subject"); 

08 while($myrow=mysql fetch array(Sresult)){ 1/ 循环 输出 投票 类 别 
09 ?> 

10 <tr bgcolor="#ECF6FF"> 

11 <td width="39">&nbsp:</td> 

12 <td colspan="2" class="stylel"> 


13 <a href="vote.php?vote_subject=<?php echo $myrow['tb_vote_subject name']:?>"> 
14 <?php echo $myrow['tb_vote_subject_ name']:?> </a></td> 


15 <td width="523" height="50"><span class="style2">&nbsp; 

16 说 明 :&nbsp; <?php echo $myrow['tb_vote_subject_text];?> </span></td> 

17 </tr> 

18 <2php }?> 

19 2 

20 <?php }else{ 

到 echo "<script>alert(' 请 您 正确 登录 ! ');window.location.href='index.php';</script>"; 

2 

投票 主题 浏览 功能 创建 完成 后 ， 接 下 来 就 是 创建 投票 内 容 提 交 页 面 。 本 系统 投票 内 容 

的 创建 是 在 后 台 完 成 的 。 


8.3.4 投票 内 容 提 交 


投票 内 容 提交 模块 的 功能 是 : 根据 超 链接 栏目 标识 中 的 变量 值 ， 从 数据 库 中 读 取出 对 
应 主题 的 数据 ， 将 投票 的 内 容 进 行 输出 ， 然 后 创建 表单 ， 实 现 投票 的 提交 ， 最 后 输出 投票 
的 结果 ， 并 且 设 置 “ 投 票 结果 ” 超 链接 ， 用 于 对 投票 结果 进行 分 析 。 投 票 内 容 提 交 页 面 
的 运行 效果 如 图 8.13 所 示 。 
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下 面 详细 介绍 投票 内 容 提交 的 实现 过 程 。 在 vote php 文件 中 ， 整 体 上 分 为 如 下 4 个 部 分 。 
(1) 判断 用 户 是 否 正常 登录 ， 如 果 是 则 可 以 进行 登录 ， 和 否则 将 弹出 提示 信息 “请 您 | 
正确 痘 录 1"， 并 返回 到 用 户 登录 页 面 。 其 关键 代码 如 下 : | 


01 <?php 全 -| 


02 session start(); // 初 始 化 SESSION 变量 | 
03 include("conn/conn.php"); // 包 含 数据 库 连 接 文 件 Note 
04 iflisset($ SESSION['tb user']){ // 判 断 用 户 是 否 正常 登录 | 
0500 2> | 
06 a | 
07 <?php }else{ | 
08 echo "<script>alert(' 请 您 正确 登录 ! ');window.location.href='index.php';</script>"; 
09 }7> | 
(2) 根据 超 链接 中 传递 的 变量 值 ， 输 出 对 应 的 投票 主题 和 说 明 的 内 容 。 代 码 如 下 : 
01 <2php | 


02 Sresult=mysql query("select * from tb vote subject where tb vote subject name= | 
™.$_GET['vote_subject].""); | 


03 while($myrow=mysql fetch array(S$result){ // 循 环 输出 查询 结果 
04 S$tpzt=$myrow['tb_vote_subject_name']; 

05 ?> 

06 <tr bgcolor="#ECF6FF"> 

07 <td colspan="2" class="stylel"> 

08 <a href="main.php"><?php echo $myrow['tb_vote_subject_name']:?></a> 
09 </td> 

10 <td width="529" height="50"><span class="stylel">&nbsp: 说 明 :&nbsp; 

11 <?php echo $myrow['tb_vote_subject_text"];?> </span> 

12 </td> 

13 </tr> 

14 <?php 

TS 

16 ?> 


(3) 输出 投票 选项 中 属于 单 选 类 的 投票 内 容 。 在 单 选 类 的 投票 中 ， 又 进一步 分 为 图 | 
片 类 投票 和 无 图 片 类 投票 两 类 。 | 
首先 ， 判 断 投票 选项 是 否 属于 单 选 类 别 ， 如 果 属 于 单 选 类 则 继续 执行 ， 创 建 一 个 form | 
表单 ， 用 于 提交 投票 选项 ， 将 结果 提交 到 tb_vote_ok.php 文件 中 。 代 码 如 下 : | 


01 <?php | 
02 $query1="select * from tb_vote_odd where tb_vote_subject=".$_GET['vote_subject].""™"; | 
03 Sresultl=mysql query(Squery1): /| 执行 查询 语句 | 
04 $myrowl=mysql fetch array($resultl): // 获 取 查 询 结 果 | 
05 S$tb_vote type=$myrowl['tb_vote_ type]: /获取 类 型 | 
06 S$tb_vote_photo=$myrowl1['tb_vote_photo']; 1/ 获取 图 片 | 
07 if($tb_vote_type==" 单 选 "){ | 
08 ?7> | 


09 <table cellpadding="0" cellspacing="0" bordercolor="#FFFFFF" background="images/bg3.gif" | 
bgcolor="#FFFFFF"> 
10 <form name="forml" method="post" action="tb_vote_ok -php" enctype="multipart/form-data"> 
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11 <tr> 
| I <td height="25" align="center"> 单 选区 </td> 
| 13 <td height="25" align="center"> 投票 主题 (<?php echo $ GET['vote_subject]:?>)</td> 


| 14 <td align="center" ><a hre 全 "see_ vote resultphp?vote_subject=<?php echo $ GET['vote_subject]:?>"> 
A 和 se 
15 </td> 


| 16 </tr> 

vote | 
| 然后 ,判断 tb_vote_photo 字段 的 值 是 否 为 空 ， 如 果 为 空 则 说 明 是 无 图 片 投票 ， 则 执行 
| 下 面 的 操作 。 应 用 while 循环 语句 ， 输 出 投票 选项 ， 并 且 设 置 单 选 按钮 来 提交 投票 。 关 键 


代码 如 下 : 

| 01 <?php 

| 02 if($tb_vote photo==""){ // 判 断 tb_vote_photo 的 值 是 否 为 空 

| 03 $query2="select * from tb_vote_odd where tb_vote_subject=".$_GET['vote_subject].""; 
| 04 Sresult2=mysql]_query($query2); // 循 环 输出 数据 

| 05 while($myrow2=mysql fetch array($result2)){ 

| 06 ?> 

| 07 <tr> 

| 08 <td width="163" align="center"><span class="style2">&nbsp;&nbsp; 

| 09 <?php echo $myrow2["tb_vote_content"]:?></span></td> 

| 10 <td width="194"> 

| 11 <input name="tb vote id" type="radio" value="<?php echo $myrow2 
| [tb vote id'];?>"> 

| 12 <input type="hidden" name="tb vote subject” value="<?php echo $myrow2 
| ['tb_vote_subject]:?>"> 

| 13 </td> 

| 14 <td align="center"> 

| 15 <table width="224" cellpadding="0" cellspacing="0"> 

| 16 <tr> 

| 17 <td width="161"” align="center" class="style2"><?php echo S$myrow2 
| [tb_vote_content']:?></td> 

| 18 <td width="61" class="style2">&nbsp;<?php echo $myrow2 
| [tb_vote_counts'];?></td> 

| 19 </tr> 

| 20 </table> 

| ol </td> 

| 22 </t> 

| 23 <?php 

24 } 

| 2 

| 26 <tr> 

| 27 <td>&nbsp:</td> 

| 2 <td><input type="submit" name="Submit" value=" 提 交 "></td> 

| 29 <td>&nbsp:</td> 

| 30 </tr> 

| 31 </form> 

| 32 ”<?php 

| 33 Jelse{ 

| 34 ?> 
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如 果 tb_vote_photo 字段 的 值 不 为 空 , 则 说 明 是 图 片 投票 , 执行 下 面 的 操作 。 应 用 while | 
循环 语句 ， 从 数据 库 中 循环 读 取 投 票选 项 的 数据 ， 并 且 对 图 片 进行 排列 输出 。 这 里 又 创建 | 
了 一 个 form 表单 ， 用 于 提交 投票 选项 ， 同 样 将 投票 结果 提交 到 tb_vote_ok.php 文件 中 。 关 | 
键 代码 如 下 ; 


01 <?php 
02 $query2="select * from tb_vote_odd where tb_vote_subject=".$_GET['vote_subject]."™"; 


03 $result2=mysql query($query2): 


04 。 Sm=0: /定义 变量 ， 用 于 图 片 的 排列 输出 | 
05 while($myrow2=mysql fetch array($result2)){ | 
06 $m++; ”// 用 于 图 片 的 排列 输出 | 
07 2 | 
08 <table width="150" border="0" align="left" cellpadding="0" cellspacing="0"> | 
09 <tr> | 
10 <td height="25" align="center"> | 


11 <table border="]l" cellpadding="0" cellspacing="0" bordercolor="#FFFFFF" bgcolor= 
"#FFFFFF"> 
12 <fonm name="form?2" method="post" action="tb_vote_ok.php" enctype= "multipart/form-data"> 


13 <tr> | 
14 <td colspan="2" align="center" valign="bottom" bgcolor="#F3F3F3"> | 
15 <img src="<?php echo substr($myrow2['tb_vote photo'],3,200);?>" width="100" | 
height="80" ></td> | 
16 </tr> | 
17 <t> | 
18 <td height="20" colspan="2" align="center" bgcolor="#F3F3F3"> | 
19 <input name="tb vote id" type="hidden" value="<?php echo $myrow2 | 
[tb_vote_ id];?>"> | 
20 <?php echo Smyrow2['tb_vote_content]:?> | 
21 <input type="hidden" name="tb vote subject" value="<?php echo $myrow2 | 
['tb_vote_subject]:?>"> | 
22 <input type="hidden" name="Submit" value=" 提 交 "> | 
23 </td> | 
24 </tr> | 
2 <tr> | 
26 <td height="30" align="center" bgcolor="#F3F3F3"> | 
27 <input name="imageField" type="image" id="imageField" src= | 
"images/].gif'></td> 
28 <td height="24" align="center" bgcolor="#F3F3F3"><a href="#"> 
29 <img src="images/2.gif' width="52" height="18" border="0" 


30 onClick="MM _ openBrWindow('see_voter_information.php?odd id=<?php echo S$myrow2 
[tb_vote_1d"];?>"," 
31 ”查看 投票 资料 ,width=300.,height=300")"></a> 


32 </td> 

33 </tr> 

34 <tr> 

35 <td height="20" col ="2" align="center" bgcolor="#F3F3F3"> 
36 <?php echo $myrow2['tb_vote_counts']:?> 

Ey </td> 

38 </tr> 

Ed </form> 
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</table> 
</td> 
</tr> 
</table> 


<?php 
计 ($m%5==0) ”// 当 变量 $m%5 的 值 等 于 0 时 ， 执 行 下 面 的 内 容 
echo "<br /><br /><br /><br /><br /><br /><br /><br /><br /><br />"; 


</td> 
<td width="1%">&nbsp:;</td> 
</tr> 
</table> 
</td> 
</tr> 
<?php } ?> 
</table> 
<?php 
} 
?> 


到 此 ， 第 (3) 部 分 输出 单 选 类 投票 内 容 介绍 完毕 。 

(4) 输出 多 选 类 投票 的 内 容 ， 应 用 的 方法 与 第 (3) 部 分 相同 ， 这 里 不 再 效 述 。 

无 论 是 单 选 或 者 多 选 的 投票 内 容 ， 都 将 提交 到 tb_vote_ok.php 文件 中 ， 将 投票 结果 添 
加 到 指定 的 数据 表 中 ， 并 通过 客户 端的 了 P 地 址 对 重复 投票 进行 限制 。 在 tb_vote_ok.php 中 
完成 投票 内 容 的 提交 可 以 分 成 如 下 几 个 步 又 : 

(1) 初始 化 Session 变量 ， 连 接 数据 库 ， 获 取 系 统 的 当前 时 间 ， 获 取 客 户 端的 P 


地 址 。 


<?php 

session_ start(); 

include("conn/conn.php"); 

$data=date("Y-m-d h:i:s"); 
$ip=getenv("REMOTE_ADDR"): // 获 取 下 


(2) 执行 单 选 投票 内 容 提 交 的 操作 ， 判 断 是 否 有 数据 提交 ， 通 过 卫 判断 是 否 是 重复 
投票 。 代 码 如 下 : 


if (isset($_POST['Submit]){ 
S$tb_vote_id=$_POST['tb_vote_id']; 
if($_POST['tb vote id]==""){ 
echo "<script>alert(' 您 没有 选择 投票 的 内 容 !"); 
window.location.href='vote.php?vote_subject=".$_POST['tb_vote_subject'].";</script>"; 
jelsef 


07 $query="select * from tb_ vote _odd ip where tb_vote_subject=".$ POST['tb_ vote _subject]." and 
tb_vote ip='$ip": 


08 


Sresult-mysql query($query): 
$row=mysql num rows(S$resulb): 
让 (Srow>0){ // 判 断 此 下 是否 重 复 

echo "<script>alert(".$ip."” 您 不 可 以 重复 投票 !"); 
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12 window.location.href='vote.php?vote_subject=".$_ POST['tb_ vote _subject].":</scripf>": 
13 }else{ 


(3) 执行 单 选 投票 的 数据 更 新 操作 ， 并 将 客户 端的 IP 地 址 存储 到 指定 的 数据 表 中 。 

代码 如 下 : 

01 $querys="update tb vote odd set tb _ vote counts=tb_ vote countstl where tb_vote id= 
'$tb_vote id"; 

02 $result=mysql query($querys); 

03 $query=mysql query("select * from tb_vote_odd where tb_vote id='$tb_vote 1d"); 

04 $myrow=mysql fetch array($query); 

05 S$tb_vote_subject=$myrow['tb_vote_subject']; 

06 S$tb_vote_content=$myrow['tb_vote_content"]; 

07 $querys="insert into tb_vote_odd ip(tb_vote_subject,tb_vote_content,tb_vote ip,tb_vote_date, 
tb_vote id) 

08 values('$tb_vote_subject','$tb_vote_content','$ip','$data','$tb_vote_id")"; 

09 Sresult=mysql_query($querys): 

10 echo "<script>alert(' 投 票 提交 成 功 !"); 


11 window.location.href='vote.php?vote_subject=".$ POST['tb_vote_subject].";</script>"; 
12 } 

13 } 

14 } 


(4) 执行 多 选 投 票 内 容 的 提交 ， 并 通过 Cookie 来 限制 重复 投票 。 代 码 如 下 : 
01 if(isset($ POST['Submit2]){ 


02 if(empty($_COOKIE['vote_cookie'])=—false){ // 通 过 Cookie 限制 重复 投票 | 
03 echo "<script>alert(' 您 不 可 以 重复 投票 ， 请 在 1 小 时 后 重新 登录 ! '); | 
04 window.location.href='vote.php?vote_subject=".$_GET['more_subject']." | 
‘</script>"; | 
05 }else{ | 
06 setcookie("vote_cookie",'value',time()+3600); // 如 果 不 存 在 ， 则 创建 Cookie | 
07 $k=0; | 
08 while (list($name,$value)=each($_POST)){ | 
09 $k+=$name; | 
10 if (is_numeric($name)=——true){ | 
11 mysql query("update tb vote more set tb vote counts=tb vote countstl where | 
tb_vote id=".$name.""); | 
12 } | 
13 } | 
14 if(Sk—0){ | 
15 echo "<script>alert(' 您 不 能 投票 ):historybackO:</script>"; | 
16 exit; | 
17 } | 
18 Imysql_query("insert into tb_vote_more ip(tb_vote_subjecttb_vote ip) | 
19 values(".$_ GET['more_subject].",'$ip)"); | 
20 echo "<script>alert( 投 票 提交 成 功 !"); | 
21 window.location.href™='vote.php?vote_subject=".$_GET['more_subject]."; | 
</script>"; | 
22 } | 
2 | 
245 3 | 
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多 
| 在 执行 多 选 投 票 内 容 提 交 的 过 程 中 ， 使 用 的 是 批量 添加 技术 ， 主 要 通过 while 循环 语 
| 句 和 list() 函 数 、each() 函 数 来 完成 。 

回 each(0) 函 数 : 返回 数组 中 当前 指针 位 置 的 键 名 和 对 应 的 值 ， 并 向 前 移动 数组 指针 。 


全 内 | 键 值 对 被 返回 为 4 个 单元 的 数组 ， 键 名 为 0、1、key 和 value。 单 元 0 和 key 包含 
SR | 数组 单元 的 键 名 ，1 和 value 包含 数据 ， 如 果 内 部 指针 越过 了 数组 的 末端 ， 则 函 
Note 数 返回 False。 其 语法 格式 如 下 : 

| array each ( array array) 


其 中 ， 参 数 array 为 输入 的 数组 。 

回 listO 函 数 : 把 数组 中 的 值 赋 给 一 些 变 量 。 与 array0 函 数 类 似 ， 不 是 真正 的 函数 ， 
而 是 语言 结构 。list0 函 数 仅 能 用 于 数字 索引 的 数组 ， 且 数字 索引 从 0 开始 。 其 语 
法 格式 如 下 : 
void list ( mixed ...) 


其 中 ， 参 数 mixed 为 被 赋值 的 变量 名 称 。 


说 明 : 
| 至 此 , 投票 内 容 提交 模块 介绍 完毕 。 至 于 投票 结果 分 析 的 实现 方法 , 参见 8.5.1 与 8.5.2 
” 节 ， 这 里 不 再 殉 述 . 


8.4 投票 管理 


”8.4.1 投票 管理 概述 


投票 管理 是 后 台 主要 的 模块 ， 控 制 着 前 台 投票 主题 和 选项 以 及 投票 结果 。 投 票 管理 的 
功能 包括 : 投票 主题 、 投 票 内 容 的 添加 、 浏 览 和 删除 ， 以 及 投票 结果 的 刷新 。 投 票 管理 首 
页 的 运行 效果 如 图 8.14 所 示 。 
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图 8.14 ”投票 管理 首页 运行 效果 
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8.4.2 动态 生成 投票 主题 及 选项 


投票 系统 的 开发 ， 第 一 步 要 做 的 就 是 添加 投票 主题 和 选项 。 因 为 只 有 具备 了 投票 主题 
和 选项 才能 够 进行 投票 ， 否 则 一 切 都 没有 意义 。 投 票 主题 和 选项 除了 可 以 从 数据 库 中 直接 | / 
添加 以 外 ， 还 可 以 创建 一 个 动态 添加 投票 主题 和 选项 的 程序 ， 通 过 动态 页 实现 主题 和 选项 | 国生 
的 添加 。 动 态 添加 投票 主题 和 选项 页 面 的 运行 效果 如 图 8.15 所 示 。 | 


于 ee | 
= “Is 


吏 ， 当 前 位 百 》 怎 傈 投票 主题 


图 8.15 动态 生成 投票 主题 及 选项 


在 adminvvote_content manage.php 文件 中 ， 通 过 form 表单 将 投票 的 主题 和 选项 添加 
到 对 应 的 数据 表 中 ， 完 成 投票 主题 和 选项 的 动态 创建 。 
(1) 创建 form 表单 ， 实 现 单 选 或 者 多 选 投 票选 项 的 动态 添加 ， 并 且 支 持 图 片上 传 。 
关键 代码 如 下 : 
01 <form action="vote_content manage ok.php" method="post" enctype="multipart/form-data" | 


name="form2"> 
02 <tr bgcolor="#167BE3"> 


03 <td height="25" valign="middle" class="style1">&nbsp; 投 票 主题 ---- 
04 ”<!-- 从 数据 表 中 选择 投票 的 主题 --> 
05 <select name="tb_vote_subject" size="1" id="tb_vote_subject"> 


07 $query="select * from tb_vote_subject": 

08 $result=mysql query($query); 

09 while($myrow=mysql fetch_array($result)){ 

10 Le ! 
11 <option value="<?php echo $myrow['tb_vote_subject_name']:?>"> | 
12 <?php echo $myrow['tb_vote_subject_name']:?></option> | 
13 <2php }2> | 


06 <?php 


14 </select></td> | 
15 </t> | 
16 <tr bgcolor="#ECF6FF"> | 
17 <td height="30">&nbsp:;&nbsp;&nbsp:;&nbsp:; | 
18 <input type="submit" name="Submit" value=" 单 选 >&nbsp:&nbsp; | 
19 <input type="submit" name="Submit2" value=" 多 选 "></td> | 
20 <t> | 
\ 
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Nw 

S 
SA 
2 </form> 


(2) 将 表单 中 的 数据 提交 到 vote_content_manage_ok.php 页 ， 根 据 “ 提 交 ” 按 钮 的 值 
进行 判断 ， 执 行将 数据 添加 到 单 选 或 者 多 选 的 数据 表 中 ; 并 且 判 断 是 否 上 传 图 片 ， 如 果 有 
图 片上 传 则 将 图 片 存储 到 服务 器 指定 的 文件 夹 下 。vote_content_manage_ok.php 的 关键 代码 
如 下 : 


01 <?php 

02 session start(); 

03 include("../conn/conn.php"); 

04 ifisset($_ SESSION['admin user]) and isset($_ SESSION['admin pass'])){ 


| 05 这 (isset($ POST['Submit]) && $ POST['Submit]==" 单 选 " and S$ _FILES 
| [tb_vote_photo']['name']==""){ 
| 06 /判断 是 单 选 ， 并 且 没有 图 片上 传 

07 /执行 将 数据 添加 到 单 选 数据 表 中 


08 $query="insert into tb vote odd(tb vote subject,tb_ vote content,tb vote counts, tb vote_ 


| type,tb_vote_explain) 


| 09 values(".$_POST['tb_vote_subject].",".$_POST['tb_vote_content].",'0',".$_POST 
‘+ ['Submit].", 
| 10 ™.$_POST['tb_vote_explain].")"; 

11 Sresult=mysql_query($query); 

12 if ($result){ 


13 echo "<script>alert(' 投 票 内 容 提 交 成 功 !");window.location.href="index.php?admin_title= 管 理 投票 


| 内容 ';</script>"; 


14 }elsefecho "失败 1";} 

入 本 

16 这 (isset($_POST['Submit]) && $_ POST['Submit]--" 单 选 " 

17 and$ FILES['tb vote photo][mame']!'=""){ /判断 是 单 选 ， 并 且 有 图 片上 传 

18 $tb_vote photo=$_FILES['b_ vote photo]['size']; /获取 图 片 大 小 

19 $tb_vote photo_name=$_FILES['b_vote photo][name']: /获取 客户 端 机 器 原文 件 的 名 称 

| 20 $tb_vote_photo_type=strtolower(strstr($tb_vote_photo_name,".")); /获取 从 "." 到 最 后 的 字符 ， 并 
| 将 字符 转换 成 小 写 

2 if($tb_vote_photo>2000000){ // 判 断 图 片 是 否 超过 指定 大 小 

bd echo "<script>alert(' 对 不 起 ,您 上 传 的 图 片 超过 2M!");history.back():</script>"; 

23 jelse{ 

24 /判断 上 传 图 片 的 后 缀 名 是 否 符合 

25 if($tb_vote_photo_type!=".jpg" & $tb_ vote photo_type!=".gif' & 

26 $tb_vote_photo_type!=".bmp" & $tb_vote_photo_type!=".jpeg"){ 

D7 echo "<script>alert(' 对 不 起 ,您 上 传 的 图 片 的 格式 不 正确 !):history.back0; 
</script>" 

28 jelse{ 

29 /定义 图 片 在 服务 器 中 的 存储 路 径 

30 S$tb_vote_photo_path=". /images/photo/' date("YmdHis") mt_rand(1000000,9999999).$_FILES 
[tb_vote photo][mame']: 

31 /应 用 move_uploaded file0 函 数 ， 将 图 片 存储 到 指定 的 文件 夹 下 

EY if(move_uploaded file($ FILES['tb_ vote photo']['tmp name'],$tb_vote_ 
Photo_path){ 

33 /执行 添加 语句 ， 将 数据 添加 到 单 选 表 中 


34 $query="insert into tb vote odd(tb vote subjecttb vote content,tb vote counts,tb vote type,tb_ 
"200。 
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vote_explain,tb_vote photo) | 
values("".$ POST['tb_vote_subject].",".$ POST['tb_vote_content].",'0',".$ POST | 
['Submit].", | 


36 ".$ POST['tb_vote_explain'].",'$tb_vote_ photo path)"; 
39 $result=mysql query($query): 
38 if (Sresull){ 


39 echo "<script>alert(' 投 票 的 内 容 提交 成 功 !");window.location.href="'index.php?admin_title= 管 理 投 
票 内 容 ';</script>"; | 
40 }else{ | 
41 echo "失败 1"; | 
42 | 


45 } 

46 } 

47 if(isset($ POST['Submitz2]) && $ POST['Submit2"]=" 多 选 " 

48 and$ FILES['tb vote photo][name] 一 ""){ /判断 提交 按钮 的 值 是 多 选 ， 并 且 无 图 片上 传 | 

49 $query="insert into tb_vote more(tb_ vote subject,tb_vote content.,tb vote counts,tb vote | 
type,tb_vote_explain) | 

50 values(".$_POST['tb_vote_subject].",".$_POST['tb_vote_content].",'0', | 

51 "S$ POST['Submit2'].",".$ POST[tb vote explain'].")"; | 

52 $result=mysql_query($query); 

$3 if ($result) { | 

54 echo "<scripf>alert( 投 票 内 容 提交 成 功 !);windowlocation href-'index.php?admin title= 管 理 投票 | 
内 容 ';</script>"; | 

$5 }else{echo "失败 1";} | 

0 

57 i (isset($ POST['Submit2]) && $ POST[Submitz]--" 多 选 " and $_FILES 
[tb_vote_photo'][mame']!=""){ 

58 $tb_vote_photo=$_FILES['tb_vote_photo']['size']; 

59 $tb_vote_photo_name=$_FILES['tb_vote_photo']['mame']; /获取 客户 端 机 器 原文 件 的 | 
名 称 | 

60 $tb_vote_ photo_type=strtolower(strstr($tb_vote_photo_name,".")); /获取 从 "." 到 最 后 的 字符 ， 并 | 
将 字符 转换 成 小 写 | 


61 if($tb_vote_photo>2000000){ | 

62 echo "<script>alert(' 对 不 起 ,您 上 传 的 图 片 超过 2M!");history.back():</script>"; | 

63 jelse{ | 

64 if($tb_vote_photo_type!=".jpg" & $tb_vote_photo_type!=".gif" & $tb_vote photo | 
type!=".bmp"){ | 

65 echo "<script>alert( 对 不 起 ,您 上 传 的 图 片 的 格式 不 正确 !" 
history.backO:</scripf"; 

66 }else{ 

67 S$tb_vote_photo_path="../images/photo/".date("YmdHis").mt_rand(1000000,9999999).$ FILES 
[tb_vote photo][mame']: 

68 if(move_uploaded file($ FILES['tb_ vote photo']['tmp_ name'],Stb_ vote 
photo_path)){ 


69 $query="insert into tb vote more(tb vote subject.tb vote content,tb_ vote counts.,tb vote type, 
了 b vote_explaintb vote photo) 
70 values("".$ POST['tb_vote_subject].",".$ POST['tb_ vote_content].",'0,".$ POST 
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te%, 

| Se 

| [Submit2].", 

| 71 ™.$_POST['tb_vote_explain'].",'$tb_vote_photo path)": 
Von Sresult=mysql query($query): 
| if ($result){ 


74 echo "<scripf>alert( 投 票 内 容 提 交 成 功 !);windowJlocation href-'index.php?admin title= 管 理 投票 
内 容 ';</script>"; 

3 }else{ 

76 echo "失败 1"; 

7 } 


82 }else{ 
83 echo "<script>alert(' 您 没有 正确 登录 ! ):windowjlocation href-admin enterphp'; </script>"; 


8.4.3 ”删除 投票 主题 与 投票 内 容 


在 图 8.15 中 ， 单 击 “ 删 除 ” 按 钮 ， 即 可 删除 当前 投票 主题 和 选项 。 删 除 操作 的 执行 ， 
是 根据 指定 投票 主题 的 ID， 执行 delete 删除 语句 。 删 除 投票 主题 的 同时 ， 也 将 删除 此 主题 
下 的 选项 内 容 。 这 一 操作 是 在 delete_vote_subject.php 文件 中 执行 的 ， 其 关键 代码 如 下 : 
01 <?php 
02 session start(); 
03 include("../conn/conn.php"); 
04 iflisset($_ SESSION['admin_ user]) && isset($_GET[vote_subject])){ 


05 $query1=mysql query("delete from tb vote subject where tb_vote_subject name= 
$_GET[vote_subject]"); 

06 $query2=mysql query("delete from tb vote odd where tb vote subject= 
$_GET[vote_subject]"); 

07 Squery3=mysql query("delete fiom {tb vote odd ip where tb _vote_ subject= 
$_GET[vote_subject]"); 

08 Squery4=mysql query("delete fiom tbvote more where tb_vote subject= 
$_GET[vote_subject]"); 

09 $query5=mysql query("delete fiom {tb vote more ip where tb vote subject= 
$_GET[vote_subject]"); 

10 echo "<script>alert(' 删 除 成 功 !); window.location hre 人 'index.php?admin title= 管 理 投 
票 主题 ;</scripf>"; 

11 Jelse{ 

i echo "<script>alert( 请 您 正 确 登录 !); window.location href~'admin enter.php'; 
</script>"; 

B30 

14 ?> 


如 果 用 户 只 是 想 删除 投票 的 某 一 项 内 容 ， 可 以 在 图 8.15 中 单 击 “ 管 理 投票 内 容 ” 超 链 
接 , 在 显示 的 投票 选项 中 找到 需要 删除 的 选项 内 容 ， 单 击 “ 删 除 ” 超 链接 ， 即 可 将 其 删除 。 
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这 一 操作 是 在 delete_vote_content.php 文件 中 执行 的 ， 其 关键 代码 如 下 。 


01 <?php 

02 session start(); 

03 include("../conn/conn.php"); 

04 iflisset($_SESSION['admin user])){ 


05 iisset($_GET[vote_content])){ 

06 $query2=mysql] query("select * from tb vote odd where tb_vote content=".$_ 
GETI['vote_content].""); 

07 Smyrow=mysql] fetch array($query2); 

08 if($myrow['tb_vote_photo']==true){ 

09 unlink($myrow['tb_vote_photo"]); 

10 } 

11 $query="delete from tb_vote_odd where tb_vote_content=".$ GET['vote_content]."™"; 

12 Sresult=mysql_query($query); 

13 echo "<script>alert(' 删 除 成 功 !");window.location.href='index.php?admin title= 管 理 投 
票 内 容 ';</script>"; 

14 } 

15 iflisset($_GET['vote_contents']){ 

16 $query4=mysql query("select * from tb vote more Where tb_vote content=".$ GET 
['vote_contents].""); 

17 $myrow4=mysql fetch_array($query4):; 

18 if($myrow4['tb_vote_photo']=—=true){ 

19 unlink($myrow4['tb_vote photo]); 

20 } 

21 Squerys="delete from tb vote more where tb vote content=".$ GET 
[vote_contents'].""; 

2 Sresults=mysql_query($querys); 

好 echo "<script>alert(' 删 除 成 功 !);window.location.hre 全 "index.php?admin_title= 管 理 投 
票 内 容 ';</script>"; 

24 } 

25 jelse{ 

26 echo "<script>alert( 请 您 正确 登录 !); windowlocationhref-'admin enter.php'; 
</script>"; 

7 

28 2 


8.4.4 刷新 投票 结果 


刷新 投票 就 是 将 当前 的 投票 结果 清 零 ， 重 新 开始 投票 。 在 图 8.16 所 示 的 页 面 中 单 击 
“重新 投票 ” 超 链接 ， 将 跳 转 到 update_vote php 文件 中 ， 根 据 超 链接 传递 的 参数 值 ， 完 成 
对 指定 投票 选项 的 更 新 操作 。 
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图 8.16 投票 系统 管理 


下 面 讲解 在 投票 管理 模块 中 刷新 投票 结果 的 方法 。 其 关键 是 在 update_vote.php 文件 
中 ， 根 据 超 链接 中 传递 的 ID 值 ， 将 指定 数据 表 中 的 tb_vote_counts 字段 的 值 更 新 为 0， 间 
删除 指定 数据 表 中 对 应 的 卫 地 址 的 记录 。 关 键 代码 如 下 : 


01 <?php 

02 session start(); 

03 include("../conn/conn.php"); 

04 iflisset($_SESSION['admin user])){ 

05 iflisset($_GET['type_o0dd"]) && S$_GET[type_odd] 一 " 单 选 9{ 

06 $query=mysql query("update tb_vote_odd set tb_vote_counts='0' where tb_vote_subject=".$_GET 
[update_id]."™"); 


下 


tt 


07 $querys=mysql query("delete from tb vote odd ip where tb_vote subject=".$_ 
GET["update_id"].""); 
08 echo "<script>alert(' 刷 新 成 功 ! "); 


09 ”window.location.href='"index.php?admin title= 管 理 投 票 &&vote subjectes=".$ GET 
["update 1d'].";</script>"; 

10 } 

11 ifisset($_GET[type_ more]) && $_GET['type_more']==" 多 选 " ){ 

12 $query=mysql query("update tb_ vote more set tb_vote counts='0' where tb_vote_subject= 
".$_ GET[update 1d].""); 

13 $querys=mysql query("delete from tb_vote more ip where tb_vote_ subject=".$_ 


GET[update id].": 


14 echo "<script>alert(' 刷 新 成 功 ! "); 
15 window.location href-'indexphp?admin title= 管 理 投 票 &&vote subjectes=".$_ GET 
[mupdate 1d'].";</script>"; 


16 } 

17 jelse{ 

18 echo "<script>alert( 请 您 正确 登录 !"); windowlocation hre 全 admin enterphp':</script>"; 
hi 

20 ?> 
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8.5 技术 提炼 
8.5.1 通过 3D 饼 图 分 析 投票 结果 四 会 4 


为 了 更 好 地 展示 在 线 投 票 的 结果 ， 使 其 更 加 直观 ， 下 面 介绍 一 种 通过 3D 饼 图 来 分 析 
投票 结果 的 方法 。 其 运行 效果 如 图 8.17 所 示 。 | 


Stat Vote Result 


Note 


国 1 
Oz 
国 3 
器 4 
口 5 
国 5 


图 8.17 通过 3D 饼 图 分 析 投票 结果 


通过 3D 饼 图 分 析 投 票 结果 ， 其 关键 就 是 3D 饼 图 的 创建 和 如 何 将 投票 结果 的 数据 传 
递 到 3D 饼 图 中 。3D 饼 图 的 创建 使 用 的 是 jpgraph 类 库 。 在 讲解 饼 图 的 创建 之 前 ， 先 来 了 
解 一 下 jpgraph 类 库 。 

jpgraph 是 一 个 强大 的 绘图 组 件 ,能 根据 用 户 的 需要 绘制 任意 图 形 。 只 要 提供 数据 ， 它 
就 能 自动 调用 绘图 函数 把 处 理 的 数据 填 进去 自动 绘制 -jpgraph 类 库 提供 了 多 种 方法 类 创建 | 
各 种 统计 图 ， 包 括 折线 图 、 柱 形 图 和 人 饼 图 等 。jpgraph 是 一 个 完全 使 用 PHP 语言 编写 的 类 | 
库 ， 可 以 应 用 在 任何 PHP 环境 中 。 | 


说 明 : | 
jpgraph 需要 GD 库 的 支持 .如 果 用 户 希 望 jpgraph 类 库 仅 对 当前 站 点 有 效 ,可 将 jpgraph | 
压缩 包 下 的 src 文件 夹 中 的 全 部 文件 复制 到 网 站 所 在 目录 的 文件 夹 中 即 可 ， 在 使 用 时 调用 | 
src 文件 夹 下 的 指定 文件 即 可 。 | 


下 面 介 绍 在 see_vote_results.php 文件 中 ， 如 何 应 用 Jpgraph 创建 3D 饼 图 , 对 投票 系统 | 

多 选 投票 的 结果 进行 分 析 。 其 操作 步骤 如 下 : | 

(1) 连接 数据 库 。 | 

(2) 应 用 include_once 语句 引用 指定 的 文件 ， 包 括 jpgraphphp、jpgraph_pie php 和 | 
jpgraph pie3d.php 文件 。 其 中 jpgraph pie.php 文件 是 饼 图 对 象 所 在 的 文件 ， 
jpgraph_pie3d.php 是 3D 饼 图 PiePlot3D 对 象 所 在 的 类 文件 。 

(3) 编写 PHP 语句 ， 从 数据 库 中 统计 出 投票 数量 。 

(4) 将 获取 的 投票 数据 写 入 一 个 数组 中 。 

(5) 创建 Graph 对 象 ， 生 成 一 个 540 像素 x260 像素 大 小 的 画布 。 
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(6) 设置 标题 的 字体 以 及 图 例 的 字体 。 
(7) 定义 生成 图 像 在 画布 中 的 位 置 。 
(8) 将 绘制 的 3D 饼 图 添加 到 图 像 中 。 
(9) 最 后 输出 图 像 。 
关键 代码 如 下 : 
| 01 <?php 
| 02 include("conn/conn.php"); 
03 include ("src/jpgraph.php"): 
04 include ("src/jpgraph pie.php"): 
05 include ("src/jpgraph pie3d.php"): // 引 用 3D 饼 图 PiePlot3D 对 象 所 在 的 类 文件 
06 /统计 投票 的 总 量 
07 $query=mysql query("select sum(tb_vote_counts) as vote_gross from tb_vote_more 
08 Where tb_vote_subject=".$_GET['vote_subject].""); 
09 S$vote counts=mysql result($query,0,"vote gross"); // 获 取 总 的 投票 数量 
10 if($vote_counts>0){ 


| 11 $querys=mysql query("select * from tb vote more where tb_vote subject=".$ GET 
| [wote_ subject]." "); 
| 12 S$vote_content=array(); 
13 Sresultes=array(; 
14 while($myrow=mysql fetch array($querys){ 
15 S$vote_content[]=$myrow['tb_vote_id']; 
16 Sresultes[]=$myrow['tb_vote_counts']; // 将 获取 的 值 写 入 到 数组 中 
I } 
18 $egraph= new PieGraph(500,245,"auto"):; // 创 建 图 像 
19 S$graph->SetShadow(); // 创 建 图 像 阴 影 
20 Sgraph->tabtitle->Set('Stat Vote Result); // 输 出 标题 
21 。 $graph->tabtitle->SetFont(FF_SIMSUN, FS_BOLD.14); /设置 标题 字体 
22 $egraph->title->SetColor("darkblue"); // 定 义 标题 颜色 
23 S$graph->legend->Pos(0.1,0.2); // 控 制 注释 文字 的 位 置 
24 S$pl=new PiePlot3d($resultes): // 创 建 图 像 
2 $p1->SetTheme("sand'"): // 控 制图 像 颜色 
26 S$pl->SetCenter(0.4); // 设 置 图 像 位 置 
27 S$pl->SetSize(0.4): // 设 置 图 像 大 小 
28 S$pl->SetHeight(20): // 设 置 饼 图 高 度 
29 S$pl->SetAngle(45); 1/ 设置 图 像 倾斜 角度 
30 S$pl->Explode(array(5,40,10,30,20)); // 控 制 饼 图 的 分 割 
31 S$pl->value->SetFont(FF_SIMSUN,FS BOLD.20): // 设 置 字体 
32 Spl->SetLegends($vote_content); 
33 S$graph->Add($p1): // 添 加 数据 
34 S$graph->Stroke(); // 生 成 图 像 
535 
36 ?> 


8.5.2 通过 柱 形 图 分 析 投 票 结果 
对 投票 结果 的 分 析 ， 不 但 可 以 通过 3D 饼 图 来 完成 ， 也 可 以 使 用 柱 形 图 来 分 析 。 下 面 
| 介绍 如 何 通过 柱 形 图 对 投票 系统 单 选 投票 的 结果 进行 分 析 ， 其 运行 效果 如 图 8.18 所 示 。 
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Stat Vote Result 
图 8.18 通过 柱 形 图 分 析 投 票 结果 


创建 see_vote_result.php 文件 ， 载 入 数据 库 连 接 文件 ， 载 入 jpgraph 类 库 ， 完 成 柱 形 图 
的 创建 。 其 步骤 如 下 : 
(1) 通过 include 语句 引用 conn.php 和 jpgraph.php 文件 。 
(2) 应 用 inleude 语句 调用 jpgraph_bar.php 文件 ， 用 于 创建 BarPlot 对 象 。 | 
(3) 从 数据 库 中 读 取出 投票 总 量 ， 计 算 每 项 投票 数量 在 总 的 投票 数量 中 所 占 的 比例 ， 
并 将 计算 结果 写 入 一 个 数组 中 。 
(4) 创建 Graph 对 象 ， 生 成 一 个 600 像素 x300 像素 大 小 的 画布 。 定 义 生成 图 像 在 画 
布 中 的 位 置 ， 以 及 画布 的 阴影 、 蓝 色 背 景 等 。 
(5) 创建 一 个 矩形 对 象 BarPlot， 设 置 其 柱 形 图 的 颜色 , 在 柱 形 图 上 方 显 示 投 票数 据 ， 
并 格式 化 数据 为 整 型 。 
(6) 将 绘制 的 柱 形 图 添加 到 画布 中 。 
(7) 添加 标题 名 称 和 XX 轴 坐 标 ， 并 分 别 设置 其 字体 。 | 
(8) 最 后 输出 图 像 。 
关键 代码 如 下 : 
01 <?php 
02 include("conn/conn.php"); 
03 include ("src/ipgraph.php"): 
04 include ("src/ipgraph_bar.php"); 
05 /统计 投票 的 总 量 


06 $query=mysql query("select sum(tb_vote_counts) as vote_gross from tb_vote_odd 
07 where tb_vote_subject=".$_GET['vote_subject].""); 


08 $vote_counts=mysql]_result($query.,0,"vote_gross"); // 获 取 总 的 投票 数量 

09 if($vote_counts>0){ 

10 $querys=mysql query("select * from tb vote odd where tb vote subject=".$_GET 
[vote_subject]." "); | 

11 $vote_content=array(); | 

位 Sresultes=array(); | 

13 while($myrow=mysql fetch array($querys){ | 

14 $vote_content[]=$myrow['tb_vote_content']; | 

15 $results=number format($myrow['tb_vote_counts']/1,2); // 计 算 每 项 投票 数量 占 | 
总 投票 量 的 比例 | 

16 Sresultes[]=S$results; // 将 获取 的 值 写 入 到 数组 中 | 
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记 } 
18 $datay=$resultes: 


19 Sgraph = new Graph(650,280,auto); // 创 建 图 像 

20 $egraph->img->SetMargin(80,30,30,40); // 定 义 图像 边 框 间 距 

21 $egraph->SetScale("textint"); // 定 义 刻度 值 的 类 型 

22 Sgraph->SetShadow(O; // 设 置 图 像 阴 影 

23 S$graph->SetFrame(false); // 不 设置 图 像 的 背景 

24 $egraph->yaxis->scale->SetGrace(35); /设置 立轴 距离 顶部 的 距离 

25 Sgraph->xaxis->SetTickLabels($vote_content): // 添 加 数据 

26 S$graph->xaxis->SetFont(FF_SIMSUN,FS BOLD):; /设置 字体 

2 Sgraph->title->Set('Stat Vote Result); /定义 标题 

28 。 $graph->title->SetFont(FF FONT2,FS BOLD); /设置 标题 字体 

29 S$graph->xaxis->title->Set("Stat Vote Result"); /设置 X 轴 的 坐标 

30 S$graph->xaxis->title->SetFont(FF_FONT2,FS_BOLD):; // 定 义 字 体 

31 $bplot=new BarPlot($datay); // 创 建 图 像 

32 $bplot->SetFillColor("orange"): // 定 义 图 像 的 填充 颜色 

33 $bplot->SetWidth(0.5); // 设 置 图 像 的 大 小 

34 $bplot->SetShadow(): // 设 置 图 像 的 阴影 

35 $bplot->value->Show():; // 输 出 图 像 对 应 的 数据 值 

36 $bplot->value->SetFont(FF ARIAL,FS BOLD): // 定 义 图 像 值 的 字体 

37 $bplot->value->SetAngle(45); // 定 义 图 像 值 的 输出 角度 

38  //$bplot->value->SetFormat('$ %0.0f); // 对 输出 值 进行 格式 化 

39 $bplot->value->SetColor("black","darkred");: // 定 义 图 像 值 的 颜色 

40 $egraph->Add($bplof): // 添 加 数据 

41 S$graph->Stroke():; // 生 成 图 像 

a0 

43 ?> 
8.5.3 ”通过 Cookie 控制 重复 投票 

在 投票 系统 中 有 一 个 很 关键 的 问题 ， 就 是 重复 投票 。 应 该 通过 什么 方法 来 处 理 这 个 问 

题 ? 投票 系统 是 否 允 许 重复 投票 ? …… 对 此 解决 方案 很 多 ， 关 键 是 根据 系统 的 实际 使 用 情 


况 而 定 ， 是 想 要 控制 一 个 IP 地 址 只 进行 一 次 投票 ， 还 是 每 间隔 一 段 时 间 进 行 一 次 投票 ， 
或 者 是 在 重新 登录 时 进行 一 次 投票 。 


下 


而 介绍 一 种 通过 Cookie 来 控制 重复 投票 的 方法 。 该 方案 实现 了 每 间隔 一 段 时 间 进 行 


一 次 投票 ， 即 同一 个 人 P 地 址 可 以 进行 多 次 投票 ， 只 是 在 多 次 投票 之 间 有 一 个 间隔 的 时 间 。 
其 运行 效果 如 图 8.19 所 示 。 


图 8.19 ”通过 Cookie 控制 重复 
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第 8 章 投票 系统 模块 


通过 Cookie 控制 重复 投票 的 原理 是 : 当 用 户 登 录 到 投票 系统 时 ， 系 统 将 判断 当前 客户 | 
端的 Cookie 变量 的 值 是 否 为 空 ， 如 果 Cookie 变量 不 为 空 则 不 能 进行 投票 如果 为 空 则 可 | 
以 进行 投票 ， 并 且 通 过 setcookie() 函 数 创建 一 个 Cookie 变量 , 指定 Cookie 的 过 期 时 间 为 1 | 


小 时 ， 以 此 来 限制 当前 客户 端 在 1 小 时 之 内 不 能 重复 投票 。 在 投票 系统 的 tb_ vote okphp | 


文件 中 ， 通 过 Cookie 对 多 选 投票 进行 限制 ， 限 制 其 在 1 小 时 之 内 不 能 进行 重复 投票 。 其 
关键 代码 如 下 : 
01 if(isset($ POST[Submit2])){ 


02 if(empty($_COOKIE['vote_cookie'])==false){ // 通 过 Cookie 控制 重复 投票 

03 echo "<script>alert(' 您 不 可 以 重复 投票 ， 请 在 1 小 时 后 重新 登录 ! '); 

04 window.location.href='vote.php?vote_subject=".$_GET['more_subject"].";</script>"; 

05 jelse{ 

06 setcookie("vote_cookie",'value',time()+3600); // 如 果 不 存 在 ， 则 创建 Cookie 

07 $k=0; 

08 while (list($name,$value)=each($ POST)){ 

09 Sk+=$name; 

10 下 (is_numeric($name) 一 true){ 

11 Imysql_ query("update tb_vote_more set tb_vote counts=tb_vote countst+] where tb_vote id= 
".$name."") 

1 

13 } 

14 1f ($k=——0){ 

15 echo "<script>alert( 您 不 能 投票 ):history.back0;</script>"; 

16 exit: 

到 } 


18 mysql query("insert into tb_ vote more ip(tb_ vote subject,tb_vote ip)values(".$_GET | 


['more_subject].",'$ip)"): 
19 ”echo "<script>alert(' 投 票 提 交 成 功 !"); 
20 window.location.href='vote.php?vote_subject=".$_GET['more_subject].";</script>"; 
2 } 
220Y 


8.5.4 ”通过 IP 限制 重复 投票 
Cookie 和 Session 都 是 对 投票 的 时 间 间 隔 进行 控制 ， 确 切 地 说 没有 完全 对 如 


如 图 8.20 所 示 。 


全 127.0.0.1 您 不 可 以 重复 投票 ! 


图 8.20 通过 了 限制 重复 投票 
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E 复 投票 进 | 
行 限制 。 这 里 介绍 一 种 完全 限制 重复 投票 的 方法 ， 即 通过 IP 限制 重复 投票 。 其 运行 效果 | 


.pp 关 旨 村 半 精 角 


JP 地 址 ， 然 后 判断 在 指定 的 数据 表 中 是 否 存 在 该 瑟 地 址 ， 如 果 存 在 该 PP 地 址 ， 则 说 明 该 
| Pp 已 经 进行 过 投票 ， 提 示 “ 您 不 可 以 重复 投票 !”， 否则 ， 可 以 进行 投票 ， 并 且 将 IP 地址 记 
| 录 到 指定 的 数据 表 中 。 在 投票 系统 的 tb_vote_ok.php 文件 中 ,应 用 他 地址 对 单 选 投票 进行 
| 限制。 关键 代码 如 下 : 


01 <?php 

02 session start(); 

03 include("conn/conn.php"); 

04 $data=date("Y-m-d h:i:s"); 

05 S$ip=getenv("REMOTE ADDR"): // 获 取 下 
06 if(isset($ POST['Submit])){ 


2 
| So 
| 通过 IP 地 址 限制 重复 投票 的 原理 是 : 当 用 户 提交 投票 时 ,首先 获取 到 客户 端 计算 机 的 


oS 
=- 


S$tb_vote_id=$_ POST['tb_vote_id']; 
08 if($_ POST['tb_vote id]==""){ 
09 echo "<script>alert(' 您 没有 选择 投票 的 内 容 !"); 
10 window.location.href='vote.php?vote_subject=".$_POST['tb_vote_subject].";</script>"; 
11 }else{ 


| 12 $query="select * from tb_vote_odd ip where tb_vote_subject=".$_ POST['tb_vote_subject]." and 
| tb_vote_ip='$ip'"; 


13 Sresult=mysql_query($query); 

14 Srow=mysql_num rows($result): 

15 if ($row>0){ 1/ 判断 此 他 是 否 重 复 

16 echo "<script>alert(".$ip.” 您 不 可 以 重复 投票 !"); 

17 window.location.href='vote.php?vote_subject=".$_POST['tb_vote_subject"]."";</script>"; 

18 jelse{ 
| 19 $querys="update tb vote odd set ”tb_vote_counts=tb_vote_counts+1 where 
| tb_vote id='$tb_vote id": 
| 20 S$result=mysql query($querys): 
| 21 $query=mysql query("select * from tb vote odd where tb _vote id= 
| '$tb_vote_id"); 
| 22 Smyrow=mysql] fetch_array($query); 

23 S$tb_vote_subject=$myrow['tb_vote_subject']: 

24 S$tb_vote_content=$myrow['tb_vote_content']; 


| 25 $querys="insert into tb_vote odd ip(tb_vote_ subject,tb_vote_content,tb_vote ip,tb_vote_ 
date,tb_vote_ id ) 
26 values('$tb_vote_subject','$tb_vote_content','$ip','$data','$tb_vote_id)"; 


2 Sresult=mysql_query($querys); 
| 28 echo "<script>alert( 投 票 提交 成 功 !"); 
| 29 window.location.href='vote.php?vote_subject=".$_POST['tb_vote_subject"]."";</script>"; 
| 30 } 
| 31 } 
3000 


| 技巧: 
在 通过 名 限制 重复 投票 的 过 程 中 ,客户 端 全 x 地 址 的 获取 使 用 的 是 getenv(0) 函 数 .getenv0 
函数 主要 是 用 于 获取 环境 变量 的 值 ， 即 客户 端的 卫 地 址 。 此 外 ， 也 可 以 使 用 

| $_SERVER[REMOTE ADDR] 来 获取 客户 端的 了 地 址 。 
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留言 本 模块 
( 铝 。 自学 视频 、 源 程序 ， 配套 资源 mA9\) 


本 章 以 构建 一 个 适用 于 中 小 型 企业 的 留言 本 为 目的 ， 全 面 介绍 留言 本 的 设 
计 思 路 及 流程 。 在 介绍 留言 本 开发 的 同时 ， 还 会 对 PHP 中 一 些 常用 函数 及 编程 
技巧 进行 详细 介绍 。 这 样 ， 不 仅 可 以 使 读者 对 留言 本 的 开发 过 程 及 思路 做 到 全 
面 了 解 并 掌 担 , 而且 还 可 以 学 习 到 许多 PHP 的 编程 技巧 从 而 全 面 提高 个 人 编 
程 能 力 。 留 言 本 的 开发 过 程 虽然 较为 简单 ， 却 涵盖 了 动态 网 站 开发 的 大 部 分 经 
典 功能 模块 ， 所 以 只 要 熟练 过 握 本 章 的 内 容 ， 并 且 能 够 灵活 应 用 ， 就 可 以 开发 
出 大 型 的 网 上 论坛 、 博 客 以 及 其 他 一 些 网 络 中 较 流 行 的 网 站 。 通 过 本 章 的 学 习 ， 
读者 可 以 学 到 以 下 内 容 。 


WI 党 担 留言 本 的 设计 思路 及 流程 

WD 党 担 My5QL 数据 库 的 设计 方式 和 方法 
WI 事 担 PHP 连接 My5QL 数据 库 的 方法 
了 能 够 灵活 运用 常用 的 SQL 查询 语句 
WI 掌握 PHP 实现 数据 分 页 显示 的 方法 


WA 


&% 
9.1 留言 本 模块 概述 


会 -| 无 论 是 应 用 程序 还 是 Web 项 目 , 开发 人 员 只 有 提前 作 好 规划 并 清楚 地 了 解 其 设计 思路 
及 流程 ， 才 能 高 效 、 高 质量 地 完成 开发 。 本 节 将 从 留言 本 的 功能 概述 、 功 能 结构 等 几 方面 
| 介绍 留言 本 的 设计 思路 及 实现 过 程 。 


9.1.1 模块 概述 


开发 一 个 适合 个 人 或 小 型 企业 使 用 的 留言 本 ， 只 需 提供 添加 留言 和 删除 留言 功能 即 
| 可。 由 于 本 章 介绍 的 留言 本 是 以 适用 于 中 小 型 企业 为 出 发 点 ， 目 的 是 使 读者 全 面 掌握 留言 
| 本 的 设计 流程 及 提高 个 人 的 编程 能 力 ， 所 以 除了 具有 添加 留言 和 删除 留言 的 功能 外 ， 还 提 
| 供 了 用 户 注 册 、 用 户 登 录 、 编辑 留言 等 功能 。 本 章 所 介绍 的 留言 本 运行 效果 如 图 9.1 所 示 。 


用 户 臭 条 主题; 五 -局 和 
Ps: 
sn 可 . 
对 误 2 所 有 法 者 朋 友 们 五 一 闻 作 ? 
万 年历 全 信箱 可 T? 引 ole ; 
日 一 二 三 四 五 六 


主 是: 护 程 河 轴 即 格 出 版 
本 四 本 台 | 
0 9 和 让 
15 16 17 18 19 20 21 


22 2 24 征 吕 条 中 XA 1 全 各 局 和 即将 出 版 ， 请 关注 ? 
2 3 3 了 
图 za s 副 六 值 箱 恶 ITPt 址 委 oica ; 


最 新 留言 主题: 久 程 记 和 什么 时 个 上 市 和 7 

， 编程 词典 什么 时 [11/05/03] 1 

， 编程 词典 即 格 出 . [11/05/03] 时 

， 五 - 快 所 ioR/09 XA :请问 扩 程 词 和 什么 时 候 上 市 和 了 
H 

写 便 箱 黑 了 地 址 A0ica ; 


共有 留言 3 条 每 页 显示 3 条 第 1 页 / 共 1 页 首页 上 一 页 下 一 页 尾 页 


『 明 日 留言 本 〗 版 权 所 有 吉林 省 明日 科技 有 限 公司 ! 未 经 授权 禁止 复制 或 建立 镜 漳 ! 


Copyright © wr ningrisoft. com All Rights Reservedl 


| 
图 9.1 留言 本 运行 效果 
| 


| 在 开发 程序 前 , 先 规划 一 下 留言 本 模块 的 功能 结构 , 以 便 在 开发 程序 时 保证 思路 清晰 。 
留言 本 模块 的 功能 结构 如 图 9.2 所 示 。 
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(1: 
(@=h: SS 


图 9.2 留言 本 模块 的 功能 结构 图 | 
9.1.3 ”程序 预览 | 


留言 本 是 由 多 个 功能 模块 组 成 的 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 
列 出 几 个 典型 功能 模块 的 运行 效果 图 ， 其 他 参见 配套 资源 中 的 源 程序 。 | 
首页 运行 效果 如 图 9.3 所 示 。 | 

* | 

1 


3 ! nt 
| 这 视 所 有 计 才 朋友 们 天 一 全 
万 年 历 写 信 箱 黑 卫 地 址 仿 0ica ; 


日 一 二 三 四 五 六 | 


10 ui 
15 18 17 18 19 20 21 


22202 5858978 XA 1 编程 亲自 即 格 出 版 ， 请 关注 9 
29 30 al i : 
团 z s 到 党 信 箱 晶 I 地 址 隐 oiea ; 
最 新 生计 主要 ”六 和 启 纲 什 么 时 候 上 市 箭 ? 
， 编程 词典 什 么 时 . [11/05/03] 
， 加 各 同和 色 格 出 [11/05/03] 群 
AR 和 NA ;请问 编程 词 和 什么 时 候 上 市 科 ? 
沪 信 箱 昌 Tb 让 且 oica ; 
后 3 条 每 RE 示 3 条 第 1 页 / 共 1 页 首页 上 -页 下 -页 尾 页 


『 明 日 世 言 态 】 版 机 所 有 吉林 省 明 巨 科技 有 限 公 司 ! 未 经 授权 棕 止 基 制 或 建立 镜 失 1 
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图 9.3 留言 本 首页 
“213°。 


ee 可 .Op 关 齐 村 决 靖 和 
| 用户 注 册页 面 运行 效果 如 图 94 所 示 。 用 户 只 有 注册 成 功 后 才 可 以 获取 登录 用 户 名 和 
| 密码， 进行 登录 后 ， 才 能 执行 发 表 留言 等 操作 。 

| 用 户 注册 

全 六 | Re 碌 完 法 和 法 重 、 行 政法 规 安 入 的 ; 


《 2 ) 煽动 兢 ! 站 会 有 
《 3) 捍 适 或 者 重 曲 事实 ， 散布 语言 ， 扰乱 社会 秩序 的 


用 户 昵 称 : Fr 
密码: se 


性 别 : 车 司 
Eniligtt: Fn 
联系 电话 : Ta 

0 号码 [aaswss45 


图 9.4 用 户 注册 页 面 
发 表 留 言 页 面 运行 效果 如 图 9.5 所 示 ; 查找 留言 页 面 运行 效果 如 图 9.6 所 示 。 


发 表 留 言 查找 留 言 
aaza: Er 请 寺 笃 查找 方式 ; [ 丙 守 到 埋 昌 
主 是 :五 -快乐 
有 计 才 朋友 仙 二 一 凶 匡 东 ? | 
留言 内 容 : | 和 i 
| A noha 
发 表 ] “| 重 写 : 信条 可 本 地址 及 0ics ) 
图 9.5 发 表 留言 页 面 图 9.6 查找 留言 页 面 


9.2 数据 库 设计 


”9.2.1 数据 库 设计 


| 
| 留言 本 模块 采用 的 是 MySQL 数据 库 ， 主 要 用 于 存储 用 户 信和 4 
| 据 库 命名 为 db_ messagebook， 其 中 包含 的 数据 表 如 图 9.7 所 示 。 


| 加 服务 崔 : localhost 》 意 数据 库 : db_messagebook 


[sn 


\ 和 留言 信息 。 这 里 将 数 


| 表 操作 记录 教 留 。 类 型 整理 襄 明 
也 leaveword 围 四 加 关 面 X 1 MYSAM gb2312_chinese_ci 。 留言 信息 夫 
th_user = 6 MYSAM gb2312_chinese_ci 。 用 户 信息 表 


2 个 表 总 计 7 MySAM gb2312_chinese_ci 


图 9.7 数据库 结构 
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9.2.2 ”数据 表 设 计 


数据 库 设 计 完成 后 ， 下 面 来 看 各 个 数据 表 的 结构 和 字段 说 明 。 | 
tb _user〈 用 户 信息 表 ) 主要 用 于 存储 用 户 信息 ， 其 结构 如 图 9.8 所 示 。 | / 


四 服务 器 : localhost > 局 数据 库 : db_messagebook ， 国 表 :tb_user ! 
字段 类 型 整理 属性 Nul 贼 认 格外 吉明 1 
u nt) 否 auto_increment 自动 六 号 Note 
useme varchar(50) gb2312_chinese_ci 否 用 户 称 
truename varchar(50) gb2312_chinese_ci 否 真实 名 称 
email varchar(50) 9b2312_chinese_ci 否 邮箱 
oq varchar(20) 。 9b2312_chinese 否 将 
‘tel varchar(20) gb2312_chinese_ci 否 电话 
np varchar(10) «gb2312_chinese_o 否 了 
address varchar(250) gb2312_chinese_ci 否 地 址 
face varchar(50) ”9b2312_chinese_ql 否 头像 
regtime 。 datetime 否 发 布 时 间 
sex varchar(2) 。 9b2312_chinese_gl 否 4 别 
usertype int(2) 否 权限 
userpwd 。 varchar(50) gb2312_chinese_c 否 攻 录 密码 


图 9.8 用户 信息 表 
tb_leaveword 留言 信息 表 ) 主要 用 于 存储 用 户 留 言 信 息 ， 其 结构 如 图 9.9 所 示 。 


加 服务 器 : localhost ， 局 数据 库 : db_messagebook 》 园 表 :tb_leaveword 
字段 类 型 整理 属性 Nal 团 认 格外 训 明 
uu nrB) 香 auto_increment 留言 这 
usend nt) 否 留言 者 ia 
createtime datetme 否 留言 时 间 
ne varchar(250) 9b2312_chinese ol 否 留言 主题 
content te 95b2312_chinese ol 否 留言 内 罕 
图 9.9 留言 信息 表 
9.2.3 ”连接 数据 库 


由 于 大 部 分 页 面 都 需要 使 用 数据 库 ， 如 果 每 页 都 编写 相同 的 数据 库 连 接 代码 ， 会 显得 
十 分 繁琐 ， 所 以 本 模块 将 数据 库 连接 代码 单独 存 入 一 个 PHP 文件 conn.php 中 ， 在 需要 与 | 
数据 库 连 接 的 页 面 中 ,使 用 包含 语句 包含 conn.php 文件 即 可 。 该 模块 实现 与 数据 库 连 接 的 | 
代码 如 下 : | 


01 <?php 

02 $conn=mysql connect("localhost","root","111"): // 连 接 数 据 库 服务 器 
03 mysql select db("db_ messagebook",$conn); // 选 择 数据 库 

04 mysql query("set names gb2312"); /设置 页 面 编码 格式 
05 Fe 
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9.3.1 发 表 留 言 概述 


用 户 登录 成 功 后 ， 单 击 导航 栏 中 的 “发 表 留 言 ” 超 链接 ， 即 可 进入 发 表 留 言 页 面 ， 如 
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图 9.10 所 示 。 在 该 页 面 中 填写 留言 主题 及 留言 内 容 ， 然 后 单 击 “ 发 表 ” 按 钮 ， 即 可 发 表 留言 。 


| 
| = 
8 上 ==aSENSAno 
于 Moe 
| 今天 是 : ”2011 年 5 月 3 日 星期 一 您 当前 的 位 置 : 明日 留言 本 ?发 表 留 言 
Note 用 户 RR 
| 用 户 各 
密 码 : 留言 主题: 夸 一 块 乐 
登录 | 重 写 
一 节 快乐 ? 
万 年 历 
| 区 和 一 汪 了 
:aassy 
8 90 um 
留言 内 容 : 
上 
2 
0 
四 wn s 四 
到 
最 新 留言 
， 编程 词典 什么 时 . [11/05/03] 
， 篇 程 词典 即将 出 , [11/05/03] 发 表 重 写 
， 五 一 快乐 [11/05/03] 


『 明 日 留言 本 】 版 权 所 有 言 林 省 明日 科技 有 限 公司 1 未 经 授权 禁止 复制 或 建立 鲁 像 | 


Copyritht © rr ningrisoft. com 人 1 Rights Roserveal 


| 图 9.10 发 表 留言 页 面 
| 9.3.2 ”发表 留言 页 面 设计 


一 个 优秀 的 Web 程序 , 不 仅 应 具有 合理 的 代码 编写 规则 和 较 高 的 代码 执行 效率 , 合理 
的 页 面 设计 方式 和 美观 的 页 面 效果 也 是 不 可 缺少 的 。 为 了 保证 整个 留言 本 页 面 的 一 致 性 ， 
在 设计 页 面 时 ， 将 留言 本 的 头 部 内 容 存储 在 top.php 文件 中 ， 将 用 于 显示 版 权 信息 的 尾部 
| 内 容 存储 在 bottom.php 文件 中 。 这 样 ， 在 新 建 留言 本 的 功能 页 面 时 ， 只 需 在 页 面 最 上 方 加 
上 include_once("top.php") 和 在 页 面 最 下 方 加 上 include_once("bottom.php") 即 可 。 应 用 这 种 
页 面 设计 方式 ， 还 可 以 提高 程序 的 开发 效率 和 易 维 护 性 。 


“技巧: 
| 

为 了 提高 网 络 的 传输 速度 , 应 尽量 将 页 面 图 片 存 储 为 GIF 或 PG 格式 (这 两 种 格式 的 
图 片 具有 占用 空间 小 、 画 面 质 量 高 等 特点 )。 另 外 ， 美 观 得 体 的 页 面 效 果 也 是 开发 人 员 所 
| 必须 考虑 的 因素 之 一 。 


发 表 留言 页 面 的 设计 效果 如 图 9.11 所 示 。 
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图 9.11 发 表 留 言 页 面 的 设计 效果 
发 表 留 言 页 面 的 设计 流程 如 下 : 


(1) 应 用 include 语句 引用 顶部 Banner 广告 头 文件 top.php， 代 码 如 下 : 

<?php include("top.php"); ?> 

(2) 应 用 include 语句 引用 左 侧 功能 导航 文件 left.php， 代 码 如 下 : 

<?php include("left.php"); ?> | 

(3) 为 了 使 页 面 效果 更 丰富 多 彩 、 更 人 性 化 ， 分 别 定义 文本 框 和 按钮 的 CSS 样式 来 
制作 个 性 化 的 表单 。 文 本 框 和 按钮 的 CSS 样式 代码 如 下 : 


01 过 -六 术 六 六 六 闪闪 六 六 站 定 义 按钮 的 CSS 样式 闻 本 相机 本 本 本 水 来 本 可 可 否 -二 
02 buttoncss { 


03 font-family: "Tahoma", "宋体 "; | 
04 font-size: pt:; color: #FCC42C: | 
05 border: 1px #003399 solid: | 
06 color:006699: | 
07 BORDER-BOTTOM: #CC42C 1px solid; | 
08 BORDER-LEFT: #FCC42C 1px solid: | 
09 BORDER-RIGHT: #FCC42C 1px solid: | 
10 BORDER-TOP: #FCC42C 1px solid: | 
11 background-color: #fFFFFF: | 
12 CURSOR: hand; | 
13 font-style: normal : | 
ww | 
15 一 -本 本 本 冰冰 事主 本 术 本 本 本 定义 文 本 框 的 CSS 样式 本 本 本 本本 本 机 本本 站- | 
16 inputcss { | 
font-size: 9pt; | 
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18 font-family: "宋体 "; 
19 font-style: normal; 
20 
21 家 
发 表 留言 页 面 涉及 到 的 HTML 表单 中 的 重要 元 素 如 表 9.1 所 示 。 
表 9.1 发 表 留言 页 面 涉 及 到 的 HTML 表单 中 的 重要 元 素 


重要 属性 
method="post" 
onsubmit="return chkinput(this, 


type="text" name="title" size="40" clas 


Dame="forml" action="saveleaveword.php" 


Ey 


s="inputcss" 


Dame 一 


“发 表 ” 按 钮 | type="submit" name: 


一 


content" rows="15" cols="55" class="inputcss" 


submit" value=" 发 表 " class="buttoncss" 


9.3.3 将 用 户 留言 内 容 保存 到 数据 库 中 


| 要 实现 发 表 留言 功能 , 首先 应 建立 用 于 填写 留言 信息 的 表单 ,然后 通过 PHP 脚本 获取 
| 表单 提交 的 数据 ， 将 留言 信息 添加 到 指定 的 数据 表 中 。 在 leaveword.php 中 完成 填写 留言 
| 信息 表单 的 创建 ,在 saveleaveword.php 文件 中 获取 表单 提交 的 留言 信息 ,并 且 将 其 添加 到 
| 指定 的 数据 表 中 。 其 关键 代码 如 下 : 


01 <?php 
02 session start(); // 初 始 化 SESSION 变量 
03 include once("conn.php"); // 包 含 数据 库 连 接 文 件 
| 04 $sql=mysql query("select id from tb_user where usernc='".$_ SESSION["unc"]."",$conn); // 
”执行 查询 语句 
| 05 $info=mysql fetch_array($sqD): // 获 取 查 询 结果 
| 06 Suserid=$info['id"]:; // 获 取 用 户 也 
| 07 S$createtime=date("Y-m-d H:i:s"); 1/ 获取 系统 当前 时 间 
| 08 /执行 添加 语句 
| 09 if(mysql_query("insert into tb_leaveword(userid,createtime,title,content) 
| 10 values('$userid','$createtime',".$_POST['title].".".$_POST['content].")",$conn)){ 
| 11 echo "<script>alert(' 留 言 发 表 成 功 ! ):historyback0O:;</script>"; 
| 12 Jyelse{ 
| 13 echo "<script>alert(' 留 言 发 表 失 败 ! ");history:back0;</script>"; 
| ey; 
| 5 
94 查看 留言 
| 9.4.1 查看 留言 概述 
| 单 击 导航 栏 中 的 “查看 留言 ” 超 链接 ， 即 可 进入 查看 留言 页 面 ， 如 图 9.12 所 示 。 在 该 
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页 面 中 ， 可 以 查看 留言 主题 、 内 容 以 及 发 布 者 等 信息 ， 还 可 对 当前 登录 用 户 的 权限 进行 判 
断 ， 如果 当前 用 户 是 留言 的 发 布 者 , 那么 还 可 以 对 留言 进行 编辑 , 否则 只 能 查看 留言 内 容 。 


令 天 是 2011 年 5 月 3 日 星期 二 您 当 前 的 位 置 : 明日 留言 本 阔 首 页 


用 户 要 录 主题 :五 -快乐 [ | 
用 户 各 : ; | 
密 码 : 人 1 | 

) ; 一 节 快 乐 ? | 
| 本 记 视 所 有 读者 贡 友 们 五 一 节 志 和 | 

万 年 历 这 信箱 出 I 地 址 让 0ica ; | 

| 

日 一 三 三 目 五 六 | E23 | 
LE ! 
sg umnsw | 
15 167 108 1920 2 £ | | 
22 23 24 25 26 27 28 A ， 轧 程 词典 即 插 出 版 ， 请 关注 人 1 
和 30 3 i | | 
| | 

司 zom s 加 澡 傅 箱 黑 可 地 址 优 0ica ; | 

是 新 引言 主题 : 妨 得 司 奥 什 么 时 个 上 市 秽 ? 钙 辑 | 
， 篇 程 词典 什么 时 [11/05/03] | ! 
， 久生 和 即 格 出. [11/05/03] 群 | 
， 五 一 快乐 [11/05/03] | 

| 
| 


入 《请问 编程 司 典 什么 时 候 上 市 币 ? 
lx 上 


污 信 箱 列 IT 地 址 局 oica ; 


共有 留言 3 条 姆 页 显示 3 条 第 1 页 / 共 ! 页 首页 上 一 页 下 一 页 尾 页 


『 明 日 留言 本 」 版 权 所 有 言 林 省 明日 科技 有 限 公司 | 未 经 授权 茜 止 复制 或 建立 蚀 像 | 


图 9.12 查看 留言 页 面 
9.4.2 ”查看 留言 页 面 设计 
为 了 使 留言 者 和 留言 内 容 相互 对 应 ， 在 设计 查看 留言 页 面 时 ， 将 留言 者 的 常规 信息 动 


态 显 示 在 页 面 左 侧 ， 将 留言 内 容 显示 在 右 侧 ， 并 通过 自 定义 函数 对 用 户 留言 内 容 中 的 
HTML 标记 进行 过 滤 。 查 看 留言 页 面 的 设计 效果 如 图 9.13 所 示 。 


图 9.13 查看 留言 页 面 的 设计 效果 
查看 留言 页 面 的 设计 流程 如 下 : 
(1) 在 HTML 标记 中 嵌入 PHP 代码 ， 动 态 地 显示 用 户 常规 信息 。 代 码 如 下 : 
01 。 <img src="images/emailgif' width="45" height-"16" alt-"<?php echo Sinfol[email]:?>"/> 


02 <img src="images/ip.gif' width="55" height="16" alt="<?php echo $infol[ip]:?>"/> 
03 <img src="images/qq.gif" width="45" height="16" alt="<?php echo $infol[qq]:?>"/> 
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(2) 通过 自 定义 函数 unhtml0 实 现 用 户 留言 中 的 HIML 标记 原样 输出 。 代 码 如 下 : 
01 function unhtml($content){ 


02 S$content=htmlspecialchars($content); // 将 单 引号 、 小 于 号 、 大 于 号 等 原样 输出 
03 S$content=str_ replace(chr(13),"<br>",$content): // 原 样 输出 换行 标记 
04 S$content=str replace(chr(32),"&nbsp:;",$content); /原样 输出 空格 


_ | 05 return trim($content); 
06 1) 


9.4.3 在 页 面 中 输出 留言 信息 


| 在 index.php 文件 中 ,通过 while 语句 循环 输出 留言 信息 的 内 容 ， 并 且 应 用 分 页 技术 控 
| 制 每 页 显示 3 条 记录 。 其 关键 代码 如 下 


01 <2php 
02 $sql=mysql_query("select count(*) as total from tb_leaveword ",$conn); 
03 S$info=mysql fetch array($sql); 
04 Stotal=$info[total]; // 获 取 总 留言 条 数 
05 这 Stotal 一 0){ // 如 果 总 留言 条 数 为 0， 则 给 出 提示 
06 echo "<div align=center> 对 不 起 ， 暂 无 留言 ! </div>"; 
07 }else{ 
08 // 判 断 查 询 字符 串 page 的 值 是 否 为 空 ， 如 果 为 空 则 默认 显示 第 一 页 
09 if(!isset($_GET["page"]) || !is_numeric($ GET["page"]){ 
10 Spage=1; 
11 }else{ 
12 S$page=intval($_GET["page"]); 
13 } 
14 S$pagesize=3; // 规 定 每 页 显示 3 条 留言 
15 if($total%$pagesize—0){ // 获 取 总 页 数 
16 Spagecount=intval($total/$pagesize):; 
17 jelse{ 
18 S$pagecount=ceil($total/$pagesize); 
19 } 
20 $sql=mysql query("select * from tb_leaveword order by createtime 
21 desc limit ".($page-1)*$pagesize.".$pagesize ",$conn); 
22 while($info=mysql fetch_array($sqD){ // 通 过 while 循环 显示 所 有 留言 
23 
24 Ts 
25 ”… ”// 显 示 用 户 信息 和 用 户 留言 信息 
26 <?php 
| 27 $sqlu=mysql query("select usernc from tb user where id=".$info["userid"]." 
| ,$conn); 
| 28 Sinfou=mysql fetch_array($sqlu); 


| 29 这 $infou["usernc"]-=-$_ SESSION["unc"]){ /判断 登录 用 户 是 否 为 留言 的 发 表 者 , 如 果 是 
| 则 显示 “编辑 ”按钮 


| 30 > 
| 31 “…// 显 示 “ 编 辑 ” 按 钮 
32 。 <?php 

| 33 } 
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34 ?> 

35 &nbsp:; 

36 <2php 

37 if($_SESSION["une"]!=""){ | 

38 $sqld=mysql query("select usertype from tb user where usernc='".$ SESSION | 才 六 
["unc"]."",$conn) | 二 一 

39 Sinfod=mysql fetch_array($sqld): | 

40 if($infod["usertype"]—1){ /判断 当前 登录 用 户 是 否 为 管理 员 ,如 果 是 则 显 ”Note 
示 “ 删 除 ” 按 钮 | 

41 ?> 

42 <a href="deleteleaveword.php?id=<?php echo $info[id];?>" class="al"> 删 除 </a> 

43 <?php 

44 ， 

45 } 

46 ?> 

47 本 /显示 用 户 信息 及 用 户 留 言 信息 

48 <?php 

49 } 

50 

51 ?> 


说 明 : 
上 述 代码 中 ,实现 留言 信息 的 分 页 显示 是 通过 MySQL 数据 库 的 扩展 关键 字 limit 来 实 | 
现 的 。 该 关键 字 后 跟 两 个 参数 ， 其 中 第 一 个 参数 用 于 指定 要 显示 记录 的 起 始 位 置 ， 而 第 二 
个 参数 用 于 指定 所 要 显示 的 记录 条 数 。 | 


9.4.4 将 留言 信息 进行 分 页 显示 


发 表 完 留言 后 ， 可 以 通过 查看 留言 页 面 查看 所 有 的 用 户 留言 内 容 。 由 于 用 户 的 留言 数 | 
目 较 多 ， ad 页 面 中 显示 所 有 留言 信息 ， 会 给 用 户 浏览 带 来 很 大 的 不 便 ， 所 以 通过 | 
分 页 的 方式 显示 用 户 留 言 内 容 是 个 不 错 的 选择 。 在 实现 用 户 留言 内 容 分 页 显示 时 ， 主 要 应 | 
用 is_numeric0 函 数 判 断 用 户 通过 GET 方法 提交 的 数据 是 否 为 数值 型 , 并 通过 ceil0 函 数 对 | 
页 码 数据 进行 向 上 取 整 。 下 面 将 对 这 两 个 函数 进行 介绍 。 | 


1. is_numericO) 函 数 | 
如 果 该 函数 的 参数 为 数字 或 数字 字符 串 则 返回 tue, 否则 返回 false。 其 语法 格式 如 下 : | 
boolis mmene Cmixed var) 

中 ， 参 数 var 为 要 进行 判断 的 数据 。 

2，ceil0 函 数 

ceil0 函 数 用 于 对 浮 点 数 进行 向 上 取 整 。 其 语法 格式 如 下 : 


float ceil ( float value ) 


。221 。 Lu 


.pp 凑 齐 搞 半 精 角 


其 中 ， 参 数 value 为 要 进行 向 上 取 整 的 数据 。 


在 显示 上 


目 户 留言 时 ， 根 据 传 入 当前 页 面 的 查询 字符 串 page 的 值 来 决定 所 要 显示 的 记 


范围 ， 创 建 分 页 超 链接 。 关 键 代 码 如 下 : 


<table width="550" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="351"><div align="left"> 共 有 留言 &nbsp;<?php echo S$total:?>&nbsp; 


每 页 显示 <?php echo S$pagesize;?>&nbsp; 条 &nbsp; 第 &nbsp;<?php echo 


$page;?>&nbsp: 


05 
06 
07 
| 08 

</a>&nbsp; 
09 


页 / 共 &nbsp:<?php echo $pagecount?>&nbsp: 页 </div> 
</td> 
<td width="199"><div align="right"> 
<a href="<?php echo $_SERVER['PHP_SELF']?>?page=1" class="al"> 首 页 


<a href="<?php echo $_ SERVER["PHP_ SELF"]?>?page= 
<?php 
if($page>1) 
echo $page-l:; 
else 
echo 1; 
?> 
" class="al"> 上 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page= 
<2php 
if($page<$pagecount) 
echo $page+1; 
else 
echo $pagecount: 
?> 
"class="al"> 下 一 页 </a>&nbsp; 
<a href="<?php echo $ SERVER['PHP SELF']:?>?page=<?php echo $pagecount:;?>" 


class="al"> 尾 页 </a></div> 


26 
27 
28 


</td> 
</tr> 
</table> 
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9.5 编辑 留 


9.5.1 编辑 留言 概述 


为 了 保证 留言 者 的 留言 信息 不 被 他 人 随意 修改 ， 用 户 只 能 对 本 人 的 留言 内 容 进行 修 
改 ， 无 权 修改 他 人 的 留言 内 容 。 编 辑 留 言 页 面 的 运行 效果 如 图 9.14 所 示 。 在 该 页 面 中 单 击 
“编辑 ” 超 链接 ， 打 开 “ 编 辑 留言 ”窗口 ， 修 改 留言 内 容 后 单 击 “ 编 辑 ” 按 钮 ， 即 可 完成 
修改 留言 操作 。 


2 


直 天 是 ;201l 年 5 月 3 可 星 由 二 总 当 和 359 和 : 轴 日 守 相 六 首页 
二 月 户 但 孙 主要 :五 = 便条 多吉 
i EETEZEOODRECEEEEE 


， 编程 词 生 即 格 出 1HMOS/031 
， 五 一 局 乐 Ht/nS/09] 


NA | 请 同志 如 司 典 什么 果 能 上 而 条 9 


这 信箱 要 地 二 由 ciss ; 
去 有 外 证 3 条 站 页 显示 3 条 第 1 页 / 共 1 再 首 而 上 -而 下 -页 尾 责 


了 困 日 久 计 下 白 可 所 有 吉林 省 明日 科技 有 限 公司 | 未 经 其 冯 枉 止 复制 束 建立 恒生 | 


Ceprright © wee ningrisoft eom A Hights Loservedl 


图 9.14 编辑 留言 页 面 
9.5.2 ”编辑 留言 页 面 设计 


为 了 使 用 户 能 够 查看 到 原 留言 的 内 容 ， 在 设计 编辑 窗口 时 ， 采 用 一 个 弹出 窗口 作为 编 | 
辑 表 单 。 编 辑 留言 页 面 的 设计 效果 如 图 9.15 所 示 。 


二 司 
i 


图 9.15 编辑 留言 页 面 的 设计 效果 
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编辑 留言 页 面 的 设计 流程 如 下 

(1) 用 户 登 录 后 ， 根 据 数据 库 中 所 保存 的 用 户 信息 与 用 户 的 登录 信息 进行 比较 ， 如 
， 果 二 者 相同 则 在 其 留言 后 显示 “编辑 ” 超 链 接 。 代 码 如 下 ; 

A 01 < 


| 02 $sqlu=mysql query("select usernc from tb_user where id="".$info["userid"]."",$conn); 
Note 03 $infou=mysql fetch_array($sqlu); 
| 04 if($infou["userne"]=—$_SESSION["unc"D{ 
05 ?> 
06 <a href="javascript:openeditwindow(<?php echo $info[id];?>)" class="al"> 编 辑 </a> 
07 <?php 
08 } 
09 2 


| (2) 当 用 户 单 击 “ 编 辑 ” 超 链接 后 ， 将 通过 自 定义 的 openeditwindow() 方 法 打开 一 个 
| 新 编辑 窗口 来 编辑 留言 。 代 码 如 下 
| 01 <script language="javascript"> 
02 function openeditwindow(x){ // 定 义 openeditwindow() 方 法 打开 弹出 窗口 
| 03 window.open("editleaveword.php?id="+x,"newframe","top=100,left=200,width=450, 
| height=280,menubar=no, location=no,scrollbars=no,status=no"): 
| 04 站 
05 </script> 


编辑 留言 页 面 涉及 到 的 HTML 表单 中 的 重要 元 素 如 表 9.2 所 示 。 
表 9.2 编辑 留言 页 面 涉及 到 的 HTML 表单 中 的 重要 元 素 
名 称 | 类 型 | 含 义 重要 属性 
Dame="forml" method="post" action="<?php echo 
$_SERVER["PHP_ SELF"]:?>" onSubmit="return chkinput(this)" 


name="title" type="text" class="inputcss" size="45" 
value="<?php echo $info[title]:?>" 


forml form 表单 


title text 发 件 人 


content textarea | 收 件 人 
id hidden “| 留言 内 容 ID | type="hidden" name="id" value="<?php echo $_GET[id]:?>" 
submit submit “编辑 ”按钮 | type="submit" value=" 编 辑 " class="buttoncss" name="submit" 


| reset reset “取消 ”按钮 | type="reset" name="reset" value=" 取 消 " class="buttoncss" 


9.5.3 ”编辑 留言 内 容 功 能 实现 


为 了 保证 用 户 的 留言 内 容 不 被 他 人 私自 更 改 ， 在 具体 实现 该 功能 时 ， 采 用 每 个 用 户 只 
能 对 自己 的 留言 内 容 进 行 更 改 的 方式 。 其 实现 原理 如 下 : 首先 ， 将 标识 用 户 身份 的 session 
变量 的 值 与 数据 库 中 该 条 留言 对 应 的 留言 者 进行 比较 ， 如 果 二 者 相同 ， 则 说 明 该 条 留言 是 
| 由 当前 登录 用 户 所 发 表 的 ， 在 该 条 留言 后 显示 “编辑 ” 超 链接 。 当 用 户 单 击 “ 编 辑 ” 超 链 
| 接 后 ， 即 可 实现 留言 信息 的 更 改 ， 并 且 在 关闭 弹出 窗口 的 瞬间 ， 留 言 信息 的 显示 页 面 也 会 
自动 进行 刷新 。 


由 
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执行 编辑 留言 操作 的 editleaveword.php 文件 的 代码 如 下 : 


01 <?php 

02 include_once("conn .php"); 

03 $id=$_GET["id"]: 

04 ”// 判 断 是 否 单 击 了 “编辑 ”按钮 ， 如 果 是 则 执行 如 下 用 于 实现 用 户 留言 信息 更 改 的 代码 ， 并， 
刷新 父 窗口 和 关闭 弹出 窗口 | 


疯 
性 


05 if($_POST["submit"]!=""){ Note 

06 if(mysql query("update tb leaveword set title=".$ POST["title"].",content=".$ POST | 
["content"]." 

07 Where id=".$_ POST["id"]."",$conn)){ 

08 echo "<script>alert(' 留 言 更 改 成 功 ! ');window.opener.location.reload();window.close() 
</script>"; 

09 }else{ | 

10 echo "<script>alert(' 留 言 更 改 失败 ! '):history.back();</script>"; | 

11 站 | 

12 exit 

1300 

14 $sql=mysql_query("select * from tb_leaveword where id=".$id."",$conn); 

15 /如 果 用 户 未 单 击 “ 编 辑 ”按钮 ， 则 查询 该 用 户 的 留言 内 容 ， 并 最 终 将 未 编辑 的 留 
言 内 容 显示 在 表单 中 

16 $info=mysql fetch_array($sqD: 

x ?> 


说 明 : 
编辑 用 户 留言 时 ， 关 闭 弹 出 窗口 、 自 动 刷新 父 窗口 ， 是 通过 如 下 语句 实现 的 。 
window.opener.location.reload(); 
上 述 代 码 的 实现 原理 是 : 在 用 window.close() 语 句 关闭 弹出 窗口 前 ， 调 用 父 窗口 的 
Teload() 方 法 实现 父 窗口 的 刷新 。 


9.6 删除 留 


qu) 


9.6.1 删除 留言 概述 

如 果 用 户 发 表 不 法 留言 ， 留 言 本 管理 员 有 权 对 其 进行 删除 。 管 理 员 登录 后 将 在 每 条 留 | 
言 后 显示 “删除 ” 超 链接 ， 当 管理 员 单 击 该 超 链接 后 ， 将 弹出 一 个 对 话 框 ， 提 示 管 理 员 是 
和 否 真 正 删除 该 条 留言 。 如 果 单 击 “确定 ”按钮 则 该 条 留言 将 被 删除 ， 反 之 不 做 任何 操作 。 


说 明 : 
本 模块 的 管理 员 用 户 名 为 mr， 密 码 为 mrsoft。 


删除 留言 页 面 的 运行 效果 如 图 9.16 所 示 。 


29s 


= 明和 要 留言 本 = 


| 今天 是 : 20U1 年 5 月 3 日 星期 二 多 当前 的 位 置 : 明日 久 本 》 首 页 
| 
袋 二 | 用户 台 杂 主 本: 五- Wt 
| 用 户 各 
Note = 登 | 
> He 
| | | 有 朋友 们 五 下? 
| 万 年 历 信箱 观 本 地 址 全 0ica ; 
| 日 ~ 二 三 四 五 六 | 证 EEC 2 了 
| 人 
| 1 本 2) WEAgEREA? 
Is 8 801902 : 
a 过 Ce | 
| 29 30 31 » 
| 四 zs 一 语 信 箱 曼 IH8 址 腿 oica ; 
| 
| i 主题 辆 和 记 什么 时 候 上 市 币 7 MR 
| ， 屿 程 词典 什么 时 [11/05/03] ' 
| ， 编程 闻 抽 妈 格 册 [11/05/03] Da 
WR XA ;请 六 程 词典 什么 时 个 上 市 和 9 
辐 , 
党 信 箱 遇 Te 址 有 oiea ; 


『 明 日 留言 本 上 版 权 所 有 吉林 省 明日 科技 有 限 公司 | 未 经 授权 梦 止 复制 或 建立 镜像 | 


Copyright © vr ningrisoft. com All Rights Reservedl 


| 图 9.16 删除 留言 页 面 
”9.6.2 ”删除 留言 页 面 设计 
删除 留言 页 面 的 设计 效果 如 图 9.17 所 示 。 


| 
| 
| 
| 
| 共有 留言 3 条 每 页 显示 3 条 第 1 页 / 共 1 页 首页 上 一 页 下 一 页 尾 页 
| 
| 
| 
| 
| 
| 
| 


图 9.17 删除 留言 页 面 的 设计 效果 
”判断 当 前 登录 用 户 是 否 为 管理 员 ， 如 果 是 则 显示 “删除 ” 超 链 接 。 代 码 如 下 : 
01 <php 


02 if($_SESSION["unc"]!=""){ // 判 断 用户 是 否 登录 
03 $sqld=mysql query("select usertype from tb user where usernc=".$ SESSION 
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04 Sinfod=mysql fetch array($sqld); | 


05 if($infod["usertype"]=——1){ // 判 断 登 录用 户 是 否 为 管理 员 

06 和 ! 

07 <a href="javascript-if(window.confirm(' 确 定 删除 该 留言 信息 么 ? ')==true) | 

08 {windowJocation href='deleteleaveword.php?id=<?php echo Sinfo[id]:?>*}" class="al"> 删 除 </a> // 如 果 | 天 六 
是 则 显示 “删除 ” 超 链 接 ~ 

09 <2php 

0 } | Note 

11 } | 

12 Dg | 


9.6.3 ”删除 留言 内 容 功 能 实现 


当 管理 员 确 认 删 除 留 言 后 ， 将 进入 留言 删除 处 理 页 deleteleaveword.php 中 ， 根 据 超 链 
接 传递 的 ID 值 ， 执 行 删除 操作 。deleteleaveword.php 的 代码 如 下 : | 


01 <?php 

02 include once("conn.php"); // 包 含 数据 库 连 接 文件 

03 if(mysql_query("delete from tb_leaveword where id=".$ GET["id"]."",$conn)){ 

04 echo "<scriptf>alert(' 留 言 删除 成 功 ! ');history.back0;</script>"; 

05 Jelse{ /删除 指定 的 留言 ， 并 给 出 提示 

06 echo "<scriptf>alert(' 留 言 删除 失败 ! "):history.back0;</script>"; | 
07 } | 
08 ?> | 


9.7 查询 留 


Qu)- 


9.7.1 查询 留言 概述 


随 着 留言 本 使 用 年 限 的 增多 ， 留 言 的 数量 会 越 来 越 多 。 为 了 方便 浏览 者 的 浏览 和 快速 | 
定位 到 指定 的 留言 ， 在 制作 留言 本 时 ， 设 计 了 可 以 按 留言 主题 和 留言 内 容 进行 模糊 查找 、 | 
按 留言 者 进行 匹配 查找 的 查询 留言 功能 。 | 

查询 留言 页 面 的 运行 效果 如 图 9.18 所 示 。 | 


器 | 1 祝 所 有 读者 前 大 们 五 一 节 纪 乐 ? 


在 革 四 节 直 司 oica ; 


图 9.18 查询 留言 页 面 
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图 9.19 查询 留言 页 面 的 设计 效果 
为 了 方便 用 户 ， 设 计 了 可 以 按 多 种 方式 进行 查找 的 查询 表单 。 代 码 如 下 : 
01 <div align="center"> 请 选择 查找 方式 : 


02 <select name="type"> 

03 <option value=""> 请 选择 </option> 
04 <option value="1"> 主 题 </option> 

05 <option value="2"> 内 容 </option> 

06 <option value="3"> 留 言 者 </option> 


07 </select>&nbsp;<input type="text" name="keyword" size="25" class="inputcss">&nbsp; 
08 <input type="submit" value=" 查 询 " class="buttoncss" name="submit"> 
09 </div> 


查询 留言 页 面 涉及 到 的 HTML 表单 中 的 重要 元 素 如 表 9.3 所 示 。 
表 9.3 查询 留言 页 面 涉及 到 的 HTML 表单 中 的 重要 元 素 


name="forml" method="post" action="<?php echo 


| forml $_SERVERI["PHP_SELF"]:?> " onsubmit="return 
| chkinput_search(this)" 
| type | select 查询 类 型 name="type" 
keyword | text 查询 关键 字 type="text" name="keyword" size="25" class="inputess" 


submit submit “查询 ”按钮 | type="submit" value=" 查 询 " class="buttoncss" name="submit" 


”9.7.3 ”查询 留言 内 容 


当 用 户 按 要 求 输入 了 查找 关键 字 ， 选 择 了 查找 方式 后 ， 单 击 “ 查 询 ” 按 钮 ， 数 据 将 被 
提交 到 searchword.php 页 , 在 该 页 完成 数据 的 查询 操作 , 并 且 输 出 查询 结果 。searchword.php 
的 关键 代码 如 下 : 
| 01 <?php 
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02 if($_POST["submit"]!=""){ // 判 断 用 户 是 否 单 击 了 “查询 ”按钮 ， 如 果 是 | 
则 开始 查询 工作 | 

03 $type=$ POST["typen]; | 

04 $keyword=$ POST["keyword"]; | 

05 if(Stype—1){ // 根 据 用 户 提交 的 查询 类 型 ， 定 义 查询 语句 。 到 六 

06 $sql=mysql query("select * from tb leaveword where title like '%".$keyword. | ~ 
"9%", $conn); 

07 Yelseif($type==2){ : Note 

08 $sql=mysql_query("select * from tb_ leaveword where content like '%".Skeyword" | 
%", $conn); | 

09 }elseif(Stype=—3){ | 

10 $sql0=mysql query("select id from tb_user where usernc=".$keyword."",$conn); | 

11 Sinfo0=mysql_fetch_array($sql0); | 

12 $sql=mysql query("select * from tb leaveword where userid=".$infoO["id"]."” | 
",$conn) | 

13 } | 

14 $info=mysql fetch array($sq]l); | 

15 if($info==false){ // 判 断 是 否 查 找到 相关 内 容 , 如 果 没 查找 到 则 显示 提示 信息 | 

16 echo "<br><br><div align=center> 对 不 起 ， 没 有 查找 到 您 要 查找 的 内 容 ! </div>"; | 

17 }else{ | 

18 dof // 通 过 循环 显示 查询 结果 | 

EL | 

20 …// 显 示 查 找 内 容 | 

oi <?php | 

22 }while($info=mysql_fetch_array($sq])); | 

23 ) 

24 } | 

25 ?> | 


9.8 技术 提炼 


9.8.1 将 数据 保存 到 数据 库 中 | 
将 用 户 发 表 的 留言 保存 到 MySQL 数据 库 是 通过 SQL 语言 的 insert 语句 实现 的 。PHP | 
通过 mysql_query0 函 数 向 MySQL 数据 库 中 发 送 SQL 命令 , 由 此 可 见 , 该 函数 在 数据 库 编 | 
程 过 程 中 具有 极其 重要 的 作用 。 | 
mysql_queryO 函 数 : 用 于 向 指定 的 连接 标识 符 所 关联 的 数据 库 服务 器 中 发 送 一 条 查询 | 
语句 。 其 语法 格式 如 下 : | 
resource mysql_ query (string query [, resource link_identifier]) | 


参数 说 明 如 下 。 | 

回 “query: 必 选 参数 ， 用 于 指定 向 数据 库 中 发 送 的 查询 语句 。 | 

回 link identifier: 可 选 参数 ， 代 表 mysql_connect() 函 数 成 功 连接 数据 库 后 所 返回 的 | 
连接 标识 ;如果 省 略 了 该 参数 ， 则 使 用 最 近 一 次 与 数据 库 建立 连接 后 所 返回 的 连 
接 标 识 。 


yp 
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[> 
”9.8.2 通过 JavaScript 实现 弹出 窗口 
| 通过 JavaScript 实现 弹出 窗口 ,主要 是 通过 调用 window 对 象 的 open0 方 法 实现 的 , 并 
。 将 实现 调用 的 代码 封装 到 一 个 单独 的 自 定义 方法 中 , 最 后 通过 HTML 元 素 的 onclick0 事 件 
， 对 该 方法 进行 调用 。 
open( 方 法 的 语法 格式 如 下 : 


| open("url","name","features") 
该 方法 的 参数 说 明 如 表 9.4 所 示 。 
表 9.4 open() 方 法 的 参数 说 明 


参数 取 值 说 了 明 
可 选 参数 。 用 于 指定 弹出 新 窗口 的 URL 地 址 ， 如 果 省 略 该 参数 或 其 值 是 一 个 空 字 
呈 符 叫 ， 那 么 新 窗口 就 不 显示 任何 文档 
新 窗口 的 名 字 。 如 果 这 个 名 字 是 一 个 已 经 存在 的 窗口 ， 那 么 open0 方 法 就 不 再 创建 
SR -个 新 窗口 ， 而 只 是 返回 这 个 窗口 的 引用 。 在 这 种 情况 下 ， 参 数 features 将 被 忽略 
ee 主要 是 显示 窗口 的 一 些 特征 。 该 参数 如 果 不 设 定 ， 则 显示 这 个 窗口 的 所 有 特征 。 
features 的 常用 特征 参见 表 9.5 
表 9.5 参数 features 的 常用 特征 
名 称 用 法 
channelmode 指定 窗口 是 否 应 该 以 频道 的 方式 显示 ， 取 值 有 (yes/no 或 1/0) 
fullscreen 指定 窗口 是 否 全 屏 显示 ， 取 值 上 有 (yes/no 或 1/0) 
menubar 指定 窗口 是 否 有 菜单 栏 ， 取 值 有 (yes/no 或 /0) 
scrollbars 指定 窗口 是 否 有 水 平 及 竖 直 滚 动 条 ， 取 值 有 (yesmo 或 1/0) 
status 指定 窗口 是 否 有 状态 栏 ， 取 值 有 〈yes/no 或 110) 
| toolbar 指定 窗口 是 否 有 工具 栏 ， 取 值 有 (yes/no 或 1/0) 
| location 指定 窗口 是 否 有 地 址 栏 ， 取 值 有 (yes/no 或 1/0) 
| directions 指定 新 建 窗口 是 否 有 标准 目录 按 扭 ， 取 值 有 (yesmo 或 1/0) 
| resizable 指定 新 窗口 大 小 是 否 可 以 调整 ， 取 值 有 (yes/no 或 1/0) 
| top 以 像素 为 单位 指定 新 窗口 上 沿 与 屏幕 上 沿 的 距离 
left 以 像素 为 单位 指定 新 窗口 左 沿 与 屏幕 左 沿 的 距离 
| width 以 像素 为 单位 指定 新 窗口 的 宽度 
| height 以 像素 为 单位 指定 新 窗口 的 高 度 


9.8.3 包含 文件 函数 

| 在 开发 Web 程序 时 , 经 常会 在 不 同 的 页 面 中 实现 相同 的 功能 或 执行 相同 的 代码 。 为 了 
| 提高 程序 开发 效率 ， 可 以 将 相同 的 代码 存储 到 单独 的 文件 中 ， 然 后 用 包含 语句 在 页 面 中 包 
| 含 该 文件 即 可 。 

include_once 语句 : 用 于 在 脚本 执行 期 间 包 含 并 运行 指定 的 文件 。 其 语法 格式 如 下 : 


include_once(filename) 
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其 中 ， 参 数 flename 用 于 指定 要 包含 的 文件 。 
9.8.4 ”MySQL 数据 库 的 函数 | 

PHP 中 提供 了 一 组 操作 MySQL 数据 库 的 函数 ， 利 用 这 些 函 数 可 以 方便 地 对 MySQL | _、y 
数据 库 进行 管理 。 | 嫩 
mysql_ fetch_array() 函 数 : 获得 数据 库 中 满足 函数 mysql_query0 中 的 SQL 语句 的 记录 。 
其 返回 值 是 一 个 数组 ， 该 数组 的 下 标 可 以 是 字段 名 ， 也 可 以 是 索引 下 标 ， 数 组 元 素 的 值 是 | 
某 个 字段 的 内 容 。 该 函数 会 使 记录 指针 自动 向 下 移动 ， 当 移动 到 最 后 一 行将 返回 一 个 false | 
值 。 其 语法 格式 如 下 : | 
array mysql fetch array(int resultint[result type]) 


其 中 参数 说 明 如 表 9.6 所 示 。 
表 9.6 mysql_fetch_array() 函 数 参数 说 明 
参数 取 值 说 了 明 
result 必 选 参数 。mysql query0 函 数 向 MySQL 数据 库 发 送 查询 命令 的 返回 标识 
可 选 参数 。 如 果 该 参数 取 MYSQL BOTH， 则 mysql_fetch_array0 函 数 将 返回 一 个 
result_type 同时 包含 关联 索引 和 数字 索引 的 数组 ;如果 该 参数 取 MYSQL ASSOC， 则 返回 一 
个 关联 索引 的 数组 ， 如 果 该 参数 取 MYSQL NUM， 则 返回 一 个 数字 索引 的 数组 
说 明 : 


与 该 函数 功能 类 似 的 函数 还 有 mysql_fetch_rows()、 mysql_result()、 mysql] fetch_object() | 
等 .mysql_fetch_rowsO 函 数 返 回 的 数组 的 下 标 为 数值 索引 下 标 。 函 数 mysql_result0 有 两 个 | 
参数 ， 其 中 第 一 个 参数 也 是 mysql query() 的 返回 结果 ， 而 第 二 个 参数 可 以 是 字段 的 偏 移 量 | 
或 者 字段 名 。 一 定 要 注意 ， 它 返回 的 结果 不 是 数组 ， 而 是 MySQL 结果 集中 一 个 单元 的 内 
容 ， 函 数 mysql_fetch_object() 的 返回 结果 是 个 对 象 ， 使 用 时 只 能 通过 字段 名 来 返回 结果 。 | 
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在 线 论坛 模块 


( 铝 # 自学 视频 、 源 程序 : 配套 资源 \》mr\10\ ) 


随 着 网 络 的 飞速 发 展 ， 人 们 对 网 站 开发 的 要 求 越 来 越 高 。 从 最 初 简单 的 静 
态 页 面 ， 到 今天 日 益 普 及 的 动态 网 站 ， 无 论 是 其 功能 结构 、 组 成 元 素 ， 还 是 设 
计 方 法 、 实 现 过 程 等 ， 均 发 生 了 质 的 飞跃 。 如 今 评 价 一 个 网 站 开发 得 是 否 成 功 
的 标准 更 高 ， 不 仅 要 看 设计 的 界面 是 否 美观 ， 还 要 看 网 站 的 功能 是 否 齐 爹 。 这 
其 中 便 包 括 一 项 必 不 可 少 的 功能 在 线 论坛 。 目 前 ， 在 线 论坛 在 商业 网 站 中 
所 扮演 的 角色 极为 重要 ， 直 接 关 系 到 企业 与 用 户 、 用 户 与 用 户 之 间 能 和 否 畅通 、 
便捷 地 进行 沟通 与 交流 。 因 此 ， 能 否 开发 一 个 完整 而 美观 的 在 线 论坛 便 成 为 程 
序 员 的 必修 课程 。 在 本 章 就 对 这 门 必 修 课程 进行 详细 的 讲解 ， 以 实战 的 形式 金 
景 展示 一 个 在 线 论坛 的 开发 过 程 。 


PI 分 页 技术 

WI 网 页 访问 权限 的 设置 
MH 搜索 引擎 的 设计 

页 面 刷新 技术 


于 
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10.1 在 线 论坛 概述 
10.1.1 “模块 概述 


论坛 是 一 个 完整 网 站 必 不 可 少 的 功能 之 一 。 通 过 在 线 论坛 ， 用 户 不 人 可 以 过 自己 的 | Note 
观点 ， 而 且 还 可 以 回复 别人 的 文章 ， 从 而 达到 相互 学 习 和 交流 的 目的 。 

在 线 论坛 主要 包括 以 下 功能 : 查看 帖子 、 发 表 帖 子 、 回 复 帖 子 、 用 户 注册 登录 、 管理 | 

员 登 录 、 添 加 主题 分 类 、 删 除 主题 和 回复 信息 ， 以 及 注销 。 其 主页 运行 效果 如 图 10.1 所 示 。 


RS 所 
区 Ce 版 主 登录 | 主 是 分 类 | 刷新 页 面 | 发 布 主题 | 返回 首页 ! 
| | 
用 户 各: 密友: [至 录 ] [注册 ] 
二 一 一 名 pi 有 
' 日 期 ou 02:51:04 ! 
全 明日 科技 出 版 的 [asp] 类 图 书 专区 主要 | 
版 主 :sp SR o | 
会 明日 科技 出 版 的 [jsp] 类 图 书 专区 a ep ;5449 | 
版 主 :JsP 今日 主题 总数 : 0 | 
全 * 明 日 科技 出 版 的 [power] 类 图 书 专区 wa ad 02;54;27 | 
版 主 :清明 今日 主题 总 数 : 0 | 
外 日 期 :2011-05-06 02:52:27 | 
全 明日 科技 出 版 的 [delphi] 类 图 书 专区 RR | 
版 主 : 九 月 今日 主题 台数 0 | 
全 明日 科技 出 版 的 [visual c++] 类 图 书 专区 全 EB 其 | 02:56:47 | 
版 主 :天 空 今日 主题 总数 : 0 | 
| 
欢迎 访问 明日 科技 网 站 ! 请 用 IE 6.0 浏览 ，1024*768 为 最 佳 效果 | 
CopyRight@2006 吉林 省 明日 科技 有 限 公司 | 
| 
| 


图 10.1 在 线 论坛 主页 
10.1.2 ”功能 结构 


本 论坛 主要 实现 以 下 功能 : 一 方面 是 供用 户 使 用 的 功能 ， 包 括 用 户 的 注册 、 登 录 , 查 | 
看 专区 类 别 ， 查 看 专区 的 帖子 主题 ， 查 看 主题 帖子 的 内 容 ， 发 布 主题 帖子 和 回复 帖子 ， 另 | 
一 方面 是 供 管理 员 使 用 的 功能 ， 包 括 管理 员 的 登录 、 栏 目 管理 、 帖 子 主题 的 管理 、 回 复 帖 | 
子 的 管理 ， 以 及 注册 会 员 的 管理 。 在 线 论坛 的 功能 结构 如 图 10.2 所 示 。 | 


10.1.3 ”权限 分 配 


在 本 论坛 系统 中 ， 对 于 不 同 的 用 户 给 予 了 不 同 的 权限 。 没 有 注册 的 用 户 ， 可 以 直接 | 
访问 本 论坛 ， 浏 览 论 坛 的 专区 帖子 ， 但 是 不 能 回复 和 发 布 主题 帖子 ; 注册 的 用 户 既 可 以 | 
浏览 论坛 的 所 有 帖子 ， 又 能 够 对 帖子 进行 回复 和 发 布 帖子 ， 论 坛 管理 员 既 可 以 对 论坛 的 | 
专区 、 帖子 进行 管理 , 也 可 以 对 论坛 的 注册 用 户 进行 管理 。 在 线 论坛 的 权限 分 配 如 图 10.3 | 
所 示 。 | 
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图 10.3 “权限 分 配 规划 图 
”10.1.4 “程序 预览 


在 线 论坛 是 由 多 个 功能 模块 组 成 的 ， 下 面 列 出 几 个 典型 功能 模块 的 运行 效果 图 ， 让 读 
六 “234 。 
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者 对 本 系统 有 一 个 初步 的 了 解 和 认识 
在 线 论坛 主页 的 运行 效果 如 图 10.4 所 示 , 其 主要 功能 是 展示 论坛 中 的 帖子 专区 , 以 及 | 
每 个 专区 中 包含 帖子 的 总 数量 和 当前 发 布 帖子 的 数量 。 | 
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10.4 在 线 论坛 主页 的 运行 效果 


分 页 浏览 在 线 论坛 指定 专区 的 帖子 ， 其 中 包括 主题 、 作 者 、 回 复 次 数 、 访 问 次 数 以 及 
发 布 时 间 ， 其 运行 效果 如 图 10.5 所 示 。 


CE I | 关节 | 


Deana 2 


后 

[EE 二 下 EC ZI 

[|e Dl mofoss | 172 | solowie 

| R45。 全， 关机 上 -页 一 珊 丰 
pM, SN eo a Dott ‘mn 


10.5 浏览 论坛 中 帖子 的 主题 


查看 论坛 中 帖子 的 具体 内 容 ， 包 括 帖子 主题 的 内 容 ， 以 及 对 应 的 回复 内 容 ， 并 且 对 回 
复 内 容 进 行 分 页 输出 。 其 运行 效果 如 图 10.6 所 示 。 


| 
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图 10.6 浏览 论坛 中 帖子 内 容 | 
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发 布 帖子 的 页 面 运行 效果 如 图 10.7 所 示 。 
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图 10.7 发 布 帖子 
10.2 数据 库 设 计 


在 线 论坛 主要 是 通过 数据 库 的 支持 来 存储 大 量 的 数据 ， 因 此 数据 库 设计 的 成 败 是 该 系 
| 统 能 否 成 功 运行 的 关键 。 前 面 已 经 对 该 程序 要 实现 的 功能 进行 了 详细 的 分 析 ， 本 节 将 针对 
前 面 提 及 的 功能 设计 一 个 合理 的 数据 库 。 


10.2.1 数据库 设计 


针对 在 线 论坛 系统 的 功能 结构 ， 创 建 一 个 名 为 mr_mysql 的 数据 库 ， 其 中 包含 的 数据 
| 表 如 图 10.8 所 示 。 


图 服务 器 : localhost 、 慷 数据 库 : mr_mysql 
于 要 作 记录 数 食 和 型 a 
my 国 轴 克基 国 X MHS gb2512_chnese_di 管用 人 息 
mw 国 看 四 王刚 XX 二 NSM。 e02312_ cmsce_o。 全 册 回 下 由 了 站 容 
| ww 国名 加 革 图 XX 了 2 NM 。atin1_swedish_d 和 卫浴 坛 中 使用 的 来 情 轴 村 
| me 图 芋 加 天 内 NSAN 9b2212_chmase_d 。 存 直 让 户 9 二 基 信 息 
| mh 国名 天国 XX 5 NHSAW cb2n12_chmsse_5 。 存 侍 沦 二 四 二 区 上 到 
mm 四 国 X NNSA gb2312_dhinese_q 。 让 镑 发 布 中 了 名 由宇 
| 人吉 总计 SA anT_swedish cl 
| 
| 
| 图 10.8 数据 表 相 关 信 息 
| 
| 
| 


10.2.2 ”数据 表 设 计 


了 解 了 数据 库 的 基本 结构 ， 下 面 详细 介绍 数据 库 中 各 个 表 的 结构 。 
mr_gly 表 用 于 存储 管理 员 注册 信息 ， 其 结构 如 图 10.9 所 示 。 


加 服务 器 : localhost > 其 数据 库 : mr_mysql， 国 表 :mr_gly 
EL 医 性 要 认 加 外 前 图 
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mr_user 表 用 于 记录 用 户 注册 的 相关 信息 ， 其 结构 如 图 10.10 所 示 。 


加 腺 务 加 :localhost ， 品 数 据 库 :mr_mysql ， 号 表 ;mr_user 
Be 村 时 属性 Nun 开 认 可 外 


Note 


否 
再 
再 
再 
再 
是 
再 
否 
再 


图 10.10 mr_ user 表 


mr_hflb 表 用 于 记录 用 户 在 回复 帖子 时 的 信息 ， 包 括 帖 子 的 主题 、 内 容 、 回 复 时 间 等 ， 
其 结构 如 图 10.11 所 示 。 


[ 网 服务 圳 -localhost ， 局 数据 库 :mr mvsal 如 家 “mr-hfb 

要 也 属性 Nul 各 本 外 而 

I reo) 要 ado naemenl 上 有 

as go2312 .hneoe_o 友 问 雪 由 的 三 是 

Mer mediarted gb2312_dhinese_5 5 RP 1 
EE 时 RE: 同 

| seemame varcharQo) go2312_chiness_o 四 加 

| MM 攻 从 

EE archarBo) go2312_chinese_o 日 人 专区 


图 10.11 mr hflb 表 
mr_lttb 表 的 作用 是 记录 论坛 中 使 用 的 表情 图 标 ， 其 结构 如 图 10.12 所 示 。 


间 服务 器 .localhost ) 局 数据 库 : mr mvsal > 回 表 mr-imb 
莫 理 展 性 。 Nu 村 认 要 外 关于 


Pe 至 ao_masmenl 。。 到 二 
abs warranto) bant_awegsn_ql 要 
noakb er 是 NL 人 


图 10.12 mr lttb 表 


mr_zqfl 表 记 录 了 在 论坛 中 有 几 个 专区 ， 及 每 个 专区 的 详细 信息 ， 包 括 专区 的 图 标 、 
版 主 、 创 建 的 时 间 ， 其 结构 如 图 10.13 所 示 。 


轩 服务 器 : localhost > 怖 数据 库 : mr mysal > 固 未 :mr-: 
要 理 属性 型 认 本 外 


县 Mo 
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crsime 和 i 


图 10.13 mr zqfl 表 


mr_zqlb 表 记 录 的 是 在 发 布 主题 中 发 布 的 帖子 的 详细 信息 ， 包 括 帖子 的 主题 、 内 容 、 
发 布 时 间 和 访问 记录 等 ， 其 结构 如 图 10.14 所 示 。 


ETEETITATERTTTTTETTPEE 
Be 于 加 司 性 Nul 轩 认 要 站 条 
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| 到 此 ， 对 在 线 论 坛 系 统 的 数据 库 设 计 已 经 介绍 完毕 ， 接 下 来 的 工作 就 是 程序 的 设计 。 
”10.2.3 ”连接 数据 库 


食 煞 | 在 对 数据 库 进行 操作 前 ， 首 先 要 建立 与 数据 库 的 连接 ， 即 使 用 config.php 文件 进行 数 
， 据 库 连接 。 首 先 使 用 mysql_connect 语句 连接 本 地 服务 器 loacalhost， 用 户 名 是 root， 密 码 

Note | 是 111; 然后 使 用 mysql_select_db 语句 选择 数据 库 mr_mysql; 最 后 ， 通 过 mysql_query0 
| 函数 设置 数据 库 的 编码 格式 为 gb2312。 其 详细 代码 如 下 : 


01 

= dr esi ny et 
02 <?php 
03 Sid=mysql connect("localhost","root","111"); // 连 接 数据 库 服 务 器 
04 mysql select db("mr mysql",$id); // 选 择 数据 库 
05 mysql query("set names gb2312"); // 设 置 编码 格式 
06 ?> 

如 果 某 个 页 面 中 需要 进行 数据 库 的 操作 ， 在 页 面 的 前 台 直 接 包含 该 文件 即 可 。 例 如 : 
01 <?php 
02 。 include("config php"): /包含 数据 库 连 接 文件 
03 ?> 


10.3 用 户 注 册 


10.3.1 用 户 注册 概述 


| 在 用 户 注册 时 ， 需 要 用 户 填写 的 必 填 信息 包括 用 户 名 、 密 码 和 E-mail 等 都 用 “*” 标 
| 注 出 来 ， 只 有 这 些 必 填 信 息 添加 完整 并 正确 时 ， 才 能 进行 注册 。 在 用 户 名 填写 完成 后 ， 用 
户 可 以 通过 单 击 “ 检 测 用 户 名 ” 超 链接 ， 检 测 所 填写 的 用 户 名 是 否 被 注册 ， 如 果 所 填写 的 
| 用 户 名 已 经 注册 过 , 则 要 填写 其 他 的 用 户 名 来 注册 。 注 册页 面 的 运行 效果 如 图 10.15 所 示 。 


| 一 一 用户 注册 信息 填写 一 


用 尸 各 : 力 用 户 登 录 在 线 论坛 
的 这 行 证 ， 可 使 用 范文 反 号， 灶 
字 或 英 六 字母 、 势 字 、 下 刘 续 的 
和 组合， 长度 控制 在 4-20 个 字符 之 
所 站 全。 请 从 和 在 妆 的 外 各， 
请 亲 为 要 大 网 ， 用户 可 以 这 泣 


+ 请 设 定 在 c-zo 们 之 问 , 痘 
未 宇 码 及 确认 室友 必须 一 下 
生 号: 输入 侯 的 生 旦 ,如 果 千 的 
主 日 是 1990 年 1 有 17 富生 入 : 
1980-0T-17, 
头 像 ; 丁 兴 适 寺 头像 下 拉 列 表 杠 


nail; 请 全 写 有 六 的 prai 地 
以便 全 名 


联系 地基。 吉林 者 长 者 


EE 


图 10.15 注册 页 面 的 运行 效果 


a . 238 . 


第 10 章 在 线 和 论坛 模块 ”和 | 
10.3.2 ”检测 用 户 名 是 否 被 占用 | 


在 注册 过 程 中 ， 用 户 名 必须 是 唯一 的 ， 不 能 同时 存在 相同 的 用 户 名 ， 因 此 首先 要 对 输 | 
入 的 用 户 名 进行 检测 。 输 入 用 户 名 ， 单 击 “ 检 测 用 户 名 ” 超 链接 ， 将 调用 Javascript 脚本 ， 
将 输入 的 用 户 名 提交 到 submit checkuserphp 文件 中 进行 检测 。submit_checkuser.php 文件 | 
的 代码 如 下 : 


01 <?php 

02 include "config.php"; // 包 含 数 据 库 连接 文件 

03 iflisset($_POST['zc_usemame']) && $ POST['zc_username']!=""){// 判 断 提 交 的 值 是 否 存 在 

04 $query="select * from mr_ user where username=".$ POST['zc username']."™"; // 定义 
查询 语句 

05 $result=mysql_query($query); // 执 行 查询 语句 

06 Snn=mysql_num rows(Sresult); // 获 取 查 询 结果 | 

07 if (Snn=—=1) { // 判 断 用 户 名 是 否 被 占用 | 

08 echo"<script>alert( 用 户 名 已 经 存在 ， 不 能 注册 ):historybackO;</script>"; 

09 } else { 

10 echo"<script>alert( 恭 喜 您 ， 可 以 用 此 用 户 名 进行 注册 ):history.back0O:</script>" 

11 } 

12 3 

13 ?> 


10.3.3 用 户 注册 的 实现 


完成 注册 信息 的 填写 后 ， 通 过 JavaScript 来 判断 用 户 填写 的 信息 是 否 符合 要 求 (如 用 | 
户 名 使 用 什么 字符 、 不 可 以 超过 多 少 个 字 节 、 日 期 的 格式 等 )， 如 果 填 写 的 用 户 信息 正确 ， 
则 单 击 “ 确 认 提交 ”按钮 ， 将 数据 提交 到 数据 处 理 页 zc_ok.php 中 ， 最 终 将 数据 添加 到 指 
定 的 数据 表 中 ， 完 成 用 户 注册 的 操作 ， 并 且 将 注册 的 用 户 名 和 密码 赋 给 SESSION 变量 。 
数据 处 理 文件 zc_ok.php 的 代码 如 下 : 


01 <?php | 
02 session start(: // 初 始 化 SESSION 变量 | 
03 include "config.php"; // 包 含 数据 库 文件 | 
04 $furl=getenv("HTTP REFERER"): /获取 了 正 ! 
05 iflisset($ POST['qgrti']){ // 对 提交 按钮 进行 判断 

06 $username=$ POST['zc_username']; | 
07 $password=-$_ POST['zc_password']: | 
08 S$zsxm=$_POST['zsxm']: 

09 $sex=$ POST[sex']: 

10 $shengri=$_POST['shengri]; | 
11 S$lxdh=$_POST['Ixdh']:; | 
12 $qq-$_POST['gq; | 
13 $tp=$_POST[ICO"]; | 
14 $email-$ POST['email]: 

15 S$grzy=$_POST['grzy']; 

16 $lxdz=$_POST['Ixdz']: 

17 $query="select * from mr user where username-'$username";// 在 数据 库 中 查询 提交 
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$result=mysql query($query); 
if(mysql num rows(Sresult)>0){ /对 查询 的 记录 数 进行 判断 
echo $usermame." 已 经 被 注册 !</font>"; // 车 用 户 名 注册 过 ， 则 给 出 提示 
供 六 | echo "<meta http-equiv=\"Refresh\”content=\"3;url=$furN\">3 秒 钟 转 入 注册 页 ,请 稍 
= 全 | 等 // 跳 转 回 注册 页 


完成 用 户 注 册 后 ， 就 可 以 登录 本 论坛 发 布 和 回复 帖子 了 。 用 户 登录 模块 的 实现 比较 简 
单 ， 此 处 不 再 袭 述 ， 具 体 的 源 代码 可 以 参见 配套 资源 (文件 名 是 djm.php 和 dl_ok.php)。 


22 jelse{ 
Note | 23 Spasswords=md5($password): 
| 24 S$tp="image/tx/".$_POST[ICO"; 
| 35 $query="insert into mr_user (username,zsxm,password,sex,shengri,lxdh,qq,tx,email, 
| grzy,lxdz) 
| 26 values('$usermame','$zsxm','$passwords','$sex','$shengri','$lxdh','$qq','$tp','$email','$grzy’,'$lxdz")"; 
| 27 if(mysql_query(Squery){ 
| 28 $_SESSION['usermame']=$username; /通过 SESSION 获取 提交 的 用 户 名 
| 29 $_SESSION['password']=$password; /通过 SESSION 获取 提交 的 密码 
| 30 echo "<font class=\"red\"> 您 注册 的 信息 如 下 ! </font><br>"; 
| 31 echo "<li class=\"huise03\"> 用 户 名 :<font color=red>".$username."<br>"; 1 
| 输出 注册 的 用 户 名 
| 32 echo "<li class=\"huise03\">E-Mail:<font color=red>".$email."<br>"; I 
| 输出 注册 的 邮箱 
| EE echo "<li class=\"huise03\"><font color=red>".Susername."</font> 恭 嘉 您 注 
| 册 成 功 ! " 
| 34 echo "<meta http-equiv=\"Refresh\" content=\"3;url=index.php\">5 秒 钟 转 入 
”主页 ,请 稍 等 …; 
| 35 jelse{ 
| 36 echo "<font class=#ff0000> 注 册 失 败 !11</font>"; 
| 3 echo mysql_ error(); 
| 38 } 
| 39 } 
0 
| 41 
| 4 


10.4 ”在线 论坛 前 台 


10.4.1 帖子 浏览 功能 概述 


笔者 设计 的 这 个 在 线 论 坛 系统 ， 其 中 的 帖子 是 公开 的 ， 只 要 游客 浏览 本 网 站 ， 就 可 以 
看 到 论坛 中 所 有 的 帖子 信息 。 因 此 ， 本 论坛 设计 的 第 一 个 功能 模块 就 是 查看 帖子 的 信息 。 
它 通 过 index.php、lmzy.php 和 zthfphp 3 个 文件 来 实现 ， 分 别 可 以 浏览 到 论坛 中 都 有 什么 
| 专区、 每 个 专区 中 都 有 什么 主题 的 帖子 、 每 个 主题 帖子 的 内 容 以 及 相关 的 回复 信息 。 论 坛 
主页 index.php 的 运行 效果 如 图 10.16 所 示 。 
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版 要闻 | 三 是 从 类 | 向 而 | 发布 主 题 | 近 加 首 页 


同 户 各: EE [ER ET 


二 是 信息 分 类 


EE 科技 BE! 


C 


日 种 出 痢 交 [asp] 关 加 和 区 
疡 主 :ap 


日 科 技 出 虎 的 [iap] 闪 图 专区 
片 主 :T 严 

“明日 种 tpover] 闪 图 专区 
晤 二 :说 明 


“明日 种 吉 册 并 交 14e1pbi| 磋 可 和 5 专 区 
年 主 洲 月 


ET 
Er 


Eg: 0 


你 
全 
全 
全 


明日 科技 出 顺 的 visusl c+] 类 图 书 专区 
版 王 :天 宁 


时 :2011-05-06 00 56: 
主题 种 ;1 


SE 0 


欢 多 访问 明 所 科 抽风 站 ! 请 条 IE .0 测 后 ，102dzT68 为 最 信 交 果 
evezo05 芷 入 有 和 人 村 


图 10.16 在 线 论坛 主页 运行 效果 


10.4.2 ”分 页 显示 回复 信息 


该 模块 虽然 是 通过 3 个 文件 来 实现 的 ， 但 是 使 用 的 方式 是 相同 的 ， 下 面 以 帖子 详细 信 | 


息 浏览 为 例 ， 讲 解 其 具体 的 实现 方法 。zthfphp 页 面 如 图 10.17 所 示 。 


所 要 姑 | 主人 分 严 | 有 页 本 上 发 和 主题 | 近 站 页 


二 


次 泛 你 改 详 评论 和 辐 材 主 占 1 生生 节 其 | 
NN ee 
Enail: AveresTgag con aa 0 发 于 间 : 2011-05-0T 加 要 


sn22Mi 刘 守 队 可笑 ，4alehi 委 泊 由 对。 让 一 生理 宁 4e2xMi 入 守 仙 人 天 。 


Tanq com 0:0 发 表 于 同 : 2011-05-07 


司 以 痢 看 车 全 


ERs 1 1 4 页 记 好 :1 条 分 页; 首页 区 而 


了 有 有 生财， 请 用 匡 和 亲 和 人 昌 人 tx 
th 


帖子 详细 内 容 和 回 


图 10.17 查看 帖子 内 容 


复 内 容 分 页 浏览 功能 的 设计 思路 如 下 : 首先 连接 到 数据 库 ， 然后 根 | 


据 条 件 从 数据 库 中 查询 数据 ， 最 后 通过 while 语句 循环 输出 查询 结果 。 


在 zthfphp 文件 中 ,首先 根据 超 链 接 传递 的 ID 值 , 从 数据 库 中 查询 出 指定 帖子 的 信息 ， 
并 且 输 出 帖子 的 主题 、 发 布 人 、 帖 子 内 容 和 发 布 时 间 。 其 关键 代码 如 下 : 

01 <?php 
02 session start(); // 初 始 化 SEESION 变量 
03 include("config.php"): // 包 含 数据 库 文件 
04 ”S$page=(isset($_GET['page']))?$_GET['page'"]:"1": // 使 用 三 元 运算 符 对 变量 page 进行 判断 
05 。 $zhuti=(isset($_GET[zhuti]D))?$_GET[zhuti]:"PHP":/ 使 用 三 元 运算 符 对 提交 的 主题 进行 判断 
06 ?> 


“241. 


~ 了 tp 各 蜡 扩 志 煤 种 


SS 

07 放 .…… 省 略 部 分 表单 元 素 .…..*/ 

08 <td width="776" height="30">&nbsp;&nbsp;&nbsp;&nbsp; 主 题 :『<?php echo $zhuti;?>」</td> 
09 /省 略 部 分 表单 元 素 -4/ 

10 <?php 


11 $query="select * 人 tom mr zqlb where id=".$_GET['recid].";// 根 据 传 递 的 人 值 ， 在 mr_zqlb 表 中 查 
询 相关 数据 

12 Sresult=mysql_query($query); 

13 S$ztrow=mysql] fetch array($result); 


14 ?> 
15 <t> 
16 <?php 


17 $update="update mr_zqlb set fwjl=fwjlH+1 where id=".$_GET['recid].”; /根据 传递 的 ID 值 ， 更 新 数 
据 表 中 对 原 帖 的 访问 量 
18 $result=mysql_query($update): 


19 $query="select* from mr user where username=".$ztrow['username'].""; /查询 发 帖 人 
的 相关 信息 
20 Sresult=mysql_query($query); 
21 $row=mysql_fetch array($result); 
22 ?> 
23 <table width="136" height="145" border="0" cellpadding="0" cellspacing="0"> 
24 <tr> 
rr] <td height="24" align="center" valign="middle" class="stylel"><?Php echo 
S$ztrow['username'];?></td> 
26 </tr> 
27 <tr> 
28 <td height="64" align="center" valign="middle"> 
29 <img sre="<?php echo $row['tx'];?>"width="60" height="60"></td> 
30 </tr> 
31 <tr> 
| 32 <td height="26" align="center” valign="middle” class="stylel"> 我 是 :<?php echo 
”Srow['sex'];?> 生 </td> 
33 </tr> 
34 <tr> 
35 <td height="30" align="center" valign="middle"> 
36 <img src="images.php?recid=<?php echo $ztrow['xq'"]:?>"></td> 
37 </t> 
38 </table> 


39 <td width="643" align="left" valign="top"> 

40 <table width="100%" border="0" align="center" cellpadding="0" cellspacing="0"> 

41 <tr> 

42 <td width="80%" height="24" class="stylel">&nbsp:Email:<?php echo 
$row['email]:?>&nbsp: 

43 QQ:<?php echo $row['qq];?> &nbsp; 发 表 时 间 : <?php echo $ztrow['fbsj]:?></td> 

44 <td width="20%" class="stylel"> 

45 <a href="hftj.php?zhuti=<?php echo urlencode($zhuti);?>&recid=<?php echo 
$_GET[recid];?>"> 回 复 </a></td> 


46 </tr> 
47 <tr> 
48 <td width="505" class="style2"> <?php echo $ztrow['neirong'];?> </td> 
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49 </tr> 
50 </table> 


然后 ， 根 据 帖子 的 ID 值 从 回复 帖子 数据 表 中 查询 出 其 对 应 的 回复 帖子 ， 并 且 对 查询 
结果 进行 分 页 输出 。 其 关键 代码 如 下 : 
01 ”<?zphp 
02 Spage_size=2; 
03 ”$query="select * from mr hflb where ljid=".$_GET['recid]."";// 根 据 传递 的 ID 值 ， 查 询 | 
mr_hflb 表 中 相关 的 回复 | 
04 Sresult=mysql query($query); | 


互 


05 $message count=mysql num rows(S$result); // 获 取 回 复 的 记录 数 | 

06 这 $message_count>0){ // 如 果 获 取 的 记录 数 大 于 0， | 
则 分 页 输出 回复 内 容 | 

07 ?> | 

08 <table width="776" border="0" align="center" cellpadding="0" cellspacing="0"> | 

09 ”<?php 

10 S$page_count=ceil($message count/$page size); // 对 所 分 的 页 数 取 整 

11 Soffset=($page-1)*$page_size; /计算 下 一 页 从 第 几 条 数据 开 | 


始 循环 

12 $quer="select * from mr hflb where ljid=".$ GET[recid']." order by id desc limit 
$offset ,Spage_size"; 

13 S$resul=mysql_query($quer); 

14 while($myrow=mysql fetch_array($resul)){ 


15 ?> | 
16 <tr> | 
17 <td height="28" colspan="3" valign="middle" background="image/5.gif' class="stylel"> | 
18 &nbsp;&nbsp; 回 复 :<?php echo $Smyrow['hfzt]:?></td> | 
19 </tr> | 
20 <tr> | 
21 <td height="100" align="left" valign="top"> 


22 <table width="776" height="100" border="0" cellpadding="0" cellspacing="0" | 
background="image/7.gif"'> | 


23 <?php $query="select * from mr_user Where usemame=".$myrow[username]."":/ 获 取 回 | 

复 者 的 相关 信息 | 
24 Sresult=mysql_query(Squery); | 

2% $xq=mysql_fetch_array($result); | 

26 号 | 

多 <tr> | 

28 <td height="24" align="center" valign="middle" class="stylel"><?php echo $myrow | 
['username'];?></td> | 
29 </tr> | 

30 <tr> | 

31 <td height="64" align="center" valign="middle"> | 

32 <img src="<?php echo $xq['tx']:;?>"width="60" height="60"></td> | 

33 </tr> | 

34 <tr> | 

35 <td width="80%" height="24" class="stylel">&nbsp:Email:<?php echo $xq | 
[email];:?>&nbsp: | 
36 QQ:<?php echo $xq['qq]:?>&nbsp: 发 表 时 间 : <?php echo $myrow['hfsj']:?></td> | 
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37 <td width="20%">&nbsp:;</td> 
| 38 </tr> 
| 39 <t> 
| 40 <td colspan="2" class="style2">&nbsp;&nbsp:<?php echo $myrow [hftnr]:?2> </td> 
全 | 41 </tr> 
NA | 42 </table></td> 
| 43 </tr> 
Note | 44 </table></td> 
| 45 </t> 
| 46 <?php 
| 47 } 
| 48 ge 
| 49 </table> 
| 50 <table width="776" height="40" border="0" align="center" cellpadding="0" cellspacing="0" 
| background="image/4.gif'> 
| 51 话 .….…. 省 略 部 分 表单 元 素 .…..*/ 
| 5 <td width="45%" class="#ff0000">&nbsp:&nbsp: 页 次 : <font class="#ff0000"><?php 


| echo $page:?> 

| 2 上] </font> / <font class="#ff0000"><?php echo $page_count:?> 

| 54 ”</font> 页 记录 : <font class="#ff0000"><?php echo $message_count:;?> </font> 条 &nbsp; 
| </td> 

| $5 <td width="45%" class="#ff0000"> 

| 56 分 页 : <a href='zthfphp?zhuti=<?php echo S$zhuti; ?>&recid=<?php echo 
| $_GET[recid]:?>>&page=1> 首 页 </a> 


| 57 <?php 
| 58 if($page >= 2){ // 如 果 页 数 超过 2 页 ， 显 示 “ 上 一 页 ” 超 链 接 
| 59 Re 
| 60 <a href="zthf php?zhuti=<?php echo Szhuti: ?> 
| 61  &recid=<?php echo $ GET['recid'];?>&page=<?php echo $page-1:?>"> 上 一 页 </a> 
| 62 <?php 
| 63 } 
| 64 这 $page < $Spage_count){ ”// 如 果 页 数 在 所 分 页 数 中 ， 显 示 “ 下 一 页 ” 超 链 接 
| 65 ?> 
| 66 <a href="zthf php?zhuti=<?php echo $zhuti; ?> 
| 67 &recid=<?php echo $ GET['recid'];?>&page=<?php echo $page+1;?>"> 下 一 页 </a> 
| 68 <?php 
| 69 } 
| 70 这 $page >= $page_count){// 如 果 页 数 大 于 等 于 所 分 页 数 ， 显 示 “ 尾 页 ” 超 链 接 
| 710 > 
| 72 <a hre 全 "zthfphp?zhuti=<?php echo $zhuti; ?> 
| 1 &recid=<?php echo $ GET['recid'];?>&page=<?php echo $page_count;?>"> 尾 页 </a> 
| 74 <2php 
| 75 } 
| 76 ?> 
| 77 ”和 #.…… 省 略 部 分 表单 元 素 .…-/ 
| 78 </table> 
| 79 ”<?php 
| 80 jelse{ 
81 2> 
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82 <table width="776" height="40" border="0" align="center" cellpadding="0" cellspacing="0" background | 


="image/4.gif'> 
83 <tr> 
84 <td height="20" colspan="3"> 无 回复 </td> | 
85 </tr> | 大 
86 </table> | 全 中 
87 ”<?php 
35 | Note 
89 ?> | 

说 明 : 


在 index.php 和 lmzy.php 文件 中 ， 使 用 的 方式 基本 是 一 致 的 ， 这 里 不 再 对 它们 进行 介 | 
绍 ， 具 体 的 代码 可 以 参见 配套 资源 。 | 


10.4.3 ”帖子 的 发 布 


作为 本 论坛 的 注册 用 户 ， 不 但 可 以 浏览 论坛 中 的 所 有 帖子 ， 还 可 以 在 论坛 中 相应 的 专 | 
区 中 发 布 自己 的 帖子 和 对 其 他 的 帖子 进行 回复 。 这 两 个 模块 看 似 不 同 ， 其 实 使 用 的 技术 是 | 
同一 个 道理 ， 都 是 通过 表单 的 形式 把 要 表达 的 内 容 提交 到 数据 处 理 页 中 ， 然 后 通过 数据 处 | 
理 页 把 信息 存储 到 数据 库 中 。 这 里 以 发 布 帖子 的 模块 为 例 进行 介绍 。 先 来 看 一 下 发 布 帖子 | 
页 面 的 运行 效果 ， 如 图 10.18 所 示 。 | 


一 次 有 于 是 

二 涩 由 人 信息 == 类 8 
abcod 到， 

» 

及 Wr a 

日 

寺 是 : 廊 生 人 

an6007a con i 


Ti127.0.0.1 


玖 名 沪 癌 明日 条 抽 阿 站 } 语 用 IE .0 浏 旺 ，102ds7c8 为 最 这 效果 
CopyRizht@2005 于 林 考 有 下 科 要 有限 公司 


图 10.18 发 布 帖子 | 

在 该 页 面 中 ， 可 以 设置 要 发 布 帖子 的 类 别 、 主 题 、 表 情 图 标 和 内 容 。 此 外 ， 还 有 一 些 

看 不 到 的 隐藏 域 , 其 内 容 包括 发 帖 人 、 发 帖 时 间 等 。 其 中 的 表情 图 标 是 从 数据 库 中 获取 的 ， | 
并 且 对 它们 进行 了 整齐 的 排列 。 表 情 图 标 排列 设计 的 代码 如 下 : | 


01 be itd pe ee et one | 
02 <?php for($i=1; $i<=4; $it+) { // 如 果 变 量 <=4， 按 照 行进 行 循环 ， 输 出 行 

03 echo"<tr>"; ?> 

04 <?php if(8i—1){ | 
05 $query="select * fom mr lttb ": /从 数据 表 mr_lttb 中 获取 表情 图 标 | 
06 Slttb=mysql_query(Squery); | 
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07 } 

08 = // 设 置 变 量 $j=1 

09 while ($lttb_ row=mysql fetch_array($lttb)){ // 进 行 while 循环 输出 
10 让 (8j<=8) {// 每 行 的 表情 图 标 个 数 <=8， 输 出 如 下 内 容 
11 ?> 

12 <td height="26" align="center" valign="middle"> 


13 <input name="xq" type="radio" value="<?php echo $lttb_row['id'];?>" checked="checked"> 

14 <img src="images.php?recid=<?php echo $lttb row["id]:?>" width="16" height="16" 
border="0"></a></td> 

15 <2php } 


17 4 人 // 当 表情 图 标的 个 数 增加 时 
18 if($j 一 9){ ”break; } ”// 当 表情 图 标的 个 数 等 于 9 时 ， 跳 转 到 下 一 行 


19 } 
20 echo "</tr>"; 

21 } 

22 ?> 


正确 填写 表单 信息 后 , 单 击 “ 主 题 提交 ”按钮 , 把 信息 提交 到 数据 处 理 页 面 fbzt_ok.php 


”中 ， 最 后 存储 到 数据 库 中 。 在 fbzt_ok php 页 ， 通 过 $_POST 获取 表单 提交 的 数据 ， 将 数据 
”添加 到 指定 的 数据 表 中 。 其 关键 代码 如 下 : 


01 <?php 

02 session_start(); /初始化 SESSION 变量 

03 。 include("config php"); /包含 数 据 库 连接 文件 

04 $time=date("Y")."-".date("m")."-".date("d"); /获取 系统 当前 时 间 

05 iisset($_ SESSION[usemame]) && isset($_SESSION['password"])){// 判 断 用 户 是 否 正 常 登录 

06 iflisset($_POST['zhuti])){ /判断 主题 是 否 提交 

07 S$zhuti=htmlspecialchars($_POST['zhuti]); // 对 字符 串 进行 格式 化 

08 Sneirong=htmlspecialchars($_POST['neirong']); 

09 S$zq=$_POST['zq"]: 

10 $xq=$_POST['xq"]; 1/ 获取 系统 提交 数据 

11 Susername=$_SESSION['usermame']:; 

12 $query="insert into mr_zqlb (zq,xq,zhuti.neirong,username,fbsj) 

13 values('$zq','$xq','$zhuti','$neirong','$username','$time’)"; 

14 Sresult=mysql_query($query,$id); // 执 行 添加 语句 

15 if($result){ 

16 echo "<meta http-equiv=\"Refresh\" content=\"2;url=lmzy.php?zq=". 
urlencode($zq)."\">"; 

17 echo "发 布 成 功 "; 

18 }else{ 

19 echo "发 布 主题 失败 11"; 

20 } 

21 } 

22 iflisset($_ POST[hft"])){ 

3 S$date=date("y:m:d h:1:s"); 

24 S$hfzt=htmlspecialchars($_POST['hfzt"]): 

23 Shfnr=htmlspecialchars($_ POST['hfnr’]): 

26 $query="insert into mr_hflb (hfzt,hfnr,hfsij,usermame,ljid,zq,zhuti) 
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27 values('$hfzt','$hfnr','$date',".$_SESSION['usemame'].", 

28 ”SPOSTI POST PO 

29 $result+mysql query($query): // 送 出 一 个 query 字符 串 
30 if($result) { 

31 echo "回复 成 功 !1"; 

32 echo "<meta http-equiv=\"Refresh\" 

33 content=\"2;url=zthfphp?zhuti=".$ POST['zt]."é&recid=".$ POST [id].\">"; 
34 }else{ 

5 echo "回复 失败 0 

36 } 

37 } 

38 jelse{ 

39 echo "请 先 登录 !1"; 

4000 

41 ?> 


上 述 代码 中 不 但 对 发 布 帖子 提交 的 数据 进行 了 处 理 ， 而 且 还 包含 了 对 回复 帖子 数据 进 | 
行 处 理 的 操作 。 | 


(1) 在 设计 表单 提交 页 面 时 ， 一 定 要 注意 在 文本 域 和 按钮 名 称 中 使 用 的 字符 。 它 是 | 
表单 提交 的 关键 ， 如 果 填写 的 字符 和 在 数据 处 理 页 fbzt ok.php 中 使 用 的 不 一 致 ， 后 果 会 很 | 
严重 ， 数 据 将 不 会 被 存储 到 数据 库 中 。 

(2) 表单 中 的 元 素 名 称 与 表单 处 理 页 使 用 的 名 称 大 小 写 必须 统一 ， 否 则 在 处 理 页 将 | 
获取 不 到 提交 的 值 。 | 

(3 ) 在 数据 处 理 页 fbzt_ok.php 中 ， 在 insert 语句 后 输出 的 字段 名 称 一 定 要 和 数据 库 | 
中 的 字段 名 称 一 致 ， 否 则 也 无 法 把 数据 存储 到 数据 库 中 。 | 


10.4.4 注销 用 户 | 
作为 注册 用 户 还 拥有 最 后 一 项 权限 ， 那 就 是 注销 用 户 。 该 权限 的 操作 与 其 代码 一 样 简 
单 ， 只 需 单 击 “ 注 销 ” 按 钮 即 可 退出 登录 。zx_ok.php 文件 的 代码 如 下 : | 
01 <?php 
02 session_ start(); 
03 session_destroy(O:; 


04 header("location:index.php"); 
05 ?> 


10.5 后 台 管 理 


10.5.1 在线 论坛 后 台 管理 概述 | 
在 线 论坛 后 台 管理 的 登录 入 口 是 论 坛 头 文件 中 的 “版 主 登录 ” 超 链接 ， 通 过 它 进入 后 | 
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台 管 理 系 统 〈 用 户 名 是 mr， 密 码 是 mrsoft)。 在 后 台 管理 系统 中 提供 了 四 大 功能 ， 分 别 介 
| 绍 如 下 。 
| 栏目 管理 : 用 于 添加 论坛 中 的 专区 、 版 主 和 对 应 的 图 标 ， 同 时 可 以 删除 指定 的 专区 。 


| 加 ”主题 管理 ， 分 页 浏览 论坛 中 发 布 的 主题 和 删除 指定 的 主题 。 
| 回 ”回复 主题 管理 :分 页 浏览 论坛 中 回复 帖子 的 内 容 和 删除 指定 的 
回 “用户 管 理 ， 分 页 浏览 论坛 中 注册 的 用 户 和 删除 指定 用 户 。 
| 后 台 管 理 系统 主页 的 运行 效果 如 图 10.19 所 示 。 
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图 10.19 后 台 首 页 


| 10.5.2 通过 switch 和 include 语句 设计 后 台 首页 框架 


| 在 后 台 管理 系统 中 ， 应 用 switch 和 include 语句 设计 一 个 网 页 框架 ， 根 据 超 链接 传递 
| 的 参数 值 ， 实 现在 栏目 管理 、 主 题 管理 、 回 复 主题 管理 和 用 户 管理 功能 之 间 的 跳 转 操作 。 
其 原理 是 ， 首先， 分 别 为 不 同 的 功能 模块 创建 超 链接 ， 都 链接 到 index php 文件 ， 通 过 超 
， 链接 传递 的 参数 值 来 区 分 不 同 的 功能 模块 ， 然后， 应 用 switch 语句 ， 根 据 超 链接 中 传递 的 
”变量 值 进行 判断 ， 并 根据 不 同 的 变量 值 应 用 include 语句 调用 不 同 的 子 文件 。 在 线 论坛 后 
”人 台 首 页 index.php 文件 的 关键 代码 如 下 : 


01 <?php 
02 session start(); 1/ 初始化 SESSION 变量 
03 include("config.php"): // 包 含 数据 库 连 接 文 件 
04 $furl=getenv("http_ referer"): 1/ 获取 当前 文件 
05 Spage=(isset($_GET['page']))?$_GET['page']:"1"; // 定 义 分 页 变量 的 值 
06 Slmbs=(isset($_GET['Imbs']))?$_GET['Imbs']:""; // 定 义 栏 目标 识 传递 的 参数 值 
07 if(isset($_SESSION['user']) && isset($_SESSION['pass'])){ /判断 当前 用 户 的 权限 
08 ?> 
09 <table width="175" border="0" cellpadding="0" cellspacing="0"> 
10 <tr> 
| 1 <td height="30" ><a href="index.php?Imbs=<?php echo urlencode(' 栏 目 管理 );?>"> 
| 栏目 管理 </a></td> 
| 12 </t> 
13 <tr> 


了 。248 。 


第 10 章 在 线 伦 坛 模块 8 | T 
A | 


14 <td height="30" ><a href="index.php?Imbs=<?php echo urlencode(' 论 坛 主题 ');?>"> 主 | 
题 管理 </a></td> | 
15 </tr> | 
16 <tr> | 
17 <td height="30" ><a hre 全 "index.php?lImbs=<?php echo urlencode(' 回 复 主 题 ');?>"> 回 复 | 全 
主题 管理 </a></td> | ~ 
18 </tr> 
i A | Note 
20 <td height="30" ><a hre 全 "index.php?lmbs=<?php echo urlencode(' 用 户 管理 );?>"> 用 | 
户 管理 </a></td> | 
21 </tr> 
2 </table> 
23 <?php 
24 switch(SImbs){ // 通 过 该 语句 来 调用 相应 的 文件 
25 case "栏目 管理 ": | 
26 include "imgLphp"; /如 果 单 击 的 超 链接 等 于 栏目 管理 , 则 输出 ImglLphp 页 | 
27 break; // 如 果 不 是 ， 则 跳出 ， 继 续 其 他 的 循环 | 
28 case "论坛 主题 ": | 
29 include "ztgl.php"; | 
30 break; | 
31 case "回复 主题 ": | 
32 include "hfztgl.php"; | 
33 break; | 
34 case "用 户 管理 ": | 
35 include "hygl.php"; | 
36 break; | 
37 default: | 
38 include "Imgl.php"; | 
39 break; | 
40 1} | 
41 ?> | 
42 <!-- 省 略 部 分 代码 --> | 
43 <?php | 
44 J}else{ /判断 当前 用 户 的 权限 | 
45 echo "<meta http-equiv=\"Refresh\" content=\"2;url=admin.php\">"; | 
46 echo "<a hre 仁 \"admin.php\"> 这 里 </a>"; | 
47 } | 
下 于 是 二 | 


10.5.3 ”栏目 管理 


后 台 管 理 系统 中 提供 了 栏目 管理 、 主 题 管 理 、 回 复 主题 管理 以 及 用 户 管理 四 大 功能 。 | 

对 于 这 些 功能 ， 可 以 将 其 总 结 为 3 个 方面 的 内 容 : 一 是 上 传 文件 ; 二 是 从 数据 库 中 查询 数 | 
据 ， 完 成 数据 的 分 页 输出 ， 三 是 删除 指定 的 数据 。 | 
所 谓 上 传 就 是 通 过 表单 提交 ， 使 用 insert 语句 把 数据 添加 到 数据 库 中 ， 而 查询 和 分 页 | 
ee 则 是 使 select 语句 从 数据 库 中 根据 不 同 的 条 件 进行 查询 ， 从 而 输出 相应 的 内 
删除 操作 是 在 使 用 select 语句 查询 出 数据 的 基础 上 ,使 用 delete 语句 以 指定 的 ID 值 为 
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b> 
| 条 件 完成 的 。 
| 下 面 以 栏目 管理 为 例 讲解 这 3 个 方面 内 容 的 具体 应 用 。 首 先是 文件 上 传 ， 在 栏目 管理 
| 中 向 数据 库 中 添加 的 是 论坛 专区 数据 ， 包 括 版 主 、 所 属 专区 和 专区 对 应 的 图 标 。 这 里 将 数 
， 据 提交 到 lmglphp 文件 中 ,应 用 insert 语句 完成 数据 的 添加 操作 。 其 创建 的 HTML 表单 和 
| 数据 处 理 的 PHP 脚本 如 下 : 
JVote | 01 <table width="590" border="1" cellpadding="0" cellspacing="0"> 


| 02 <form name="myform" method="post” action="indexphp?lmbs= 栏目 管理 " 
enctype="multipart/form-data"> 


| 03 <tr> 

| 04 <td height="30" colspan="2" align="center"> 栏 目 管理 </td> 

| 05 <td width="220">&nbsp:</td> 

| 06 </tr> 

| 07 <t> 

| 08 <td width="170" height="30" align="center"><span class="stylel"> 版 主 : 
| 09 <input name="bz" type="text" id="bz" size="15"> 

| 10 </span></td> 

| 11 <td width="200" align="center" 0 属 专 区 : 

| 12 <select name="zq" size="1" id="zq 

| 13 <option value="asp" eR 

| 14 <option value="jsp">JSP</option> 

| 15 <option value="delphi">Delphi</option> 

| 16 <option value="visual basic">Visual Basic</option> 

| 7 <option value="visual foxpro">Visual Foxpro</option> 

| 18 <option value="visual c++">Visual C++ </option> 

| 19 <option value="power">Power Buider</option> 

| 20 <option value=".net">.net</option> 

| 21 </select></td> 

| 22 <td align="center" valign="middle" class="stylel"> 图 标 : 

| 23 <input type="radio" name="tb" value="<?php echo $top;?>"> 

| 24 <img sre="../image/tx/top.gif width='24' height='24> 

| 25 <input type="radio" name="tb" value="<?php echo $df:?>"> 

| 26 <img src=' image/tx/df gif width='24 height='24'> 

| 27 </td> 

| 28 </tr> 

| 29 <tr> 

| 30 <td colspan="2">&nbsp:</td> 

| 31 <td class="stylel"><input name="zhuijia" type="submit" id="zhuijia" 
”value=" 追 加 栏目 "></td> 

| 32 </tr> 

| 33 </form> 

| 34 </table> 

| 35 <?php 

| 36 iflisset($_SESSION['user']) && isset($ SESSION['pass']){ 

| 37 S$top="image/tx/top.gif : // 通 过 变量 来 获取 专区 的 图 标 
| 38 $df="image/tx/df. gif 

| 39 $date=date("Y-m-dh:i:s"): /获取 当前 的 时 间 

| 40 ifisset($_POST[zhuijia])){ // 把 提交 的 数据 添加 到 数据 库 中 
| 41 Squery="insert into mr_zqfl(bz,zq,tb,cjsj) 

| 42 values(™.$_POST['bz].",".$_POST['zq].",".$_POST['tb"].",".$date.")"; 
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43 Sresult—=mysql]_query($query); | 

4 if($resul) { | 

45 echo "<meta http-equiv=\"refresh\" content=\"0:url=$furl\">"; | 

46 jelsef | 

47 echo "追加 失败 !1"; | 1 
48 } | 凶 | 
49 } | 

50 ?> Note 


然后 ， 在 ImgLphp 文件 中 ， 连 接 数据 库 ， 通 过 select 语句 查询 数据 库 中 存储 的 论坛 专 | 

区 数据 ， 应 用 while 语句 循环 输出 查询 结果 ， 并 且 创 建 “ 删 除 ” 超 链接 ， 链 接 到 delete.php | 
文件 ， 以 论坛 专区 的 ID 值 为 参数 。 其 关键 代码 如 下 : | 
01 <table width="590" border="0" cellpadding="0" cellspacing="0"> | 


02 <tr align="center" bgcolor="#DO0E8FF" class="stylel"> 
03 <td width="95" height="30"> 图 标 </td> 

04 <td width="70"> 所 属 专区 </td> 

05 <td width="145" bgcolor="#D0E8FF"> 版 主 </td> 

06 <td width="200"> 时 间 </td> 

07 <td width="80"> 是 否 删除 </td> 

08 </tr> 

09 <?php 


10 ”$query="select* from mr zqfl where id"; 。 // 从 数据 库 中 查询 出 有 关 栏 目 专区 的 信息 
11 $result=mysql_query($query); 
12 if($result){ 


13 while($myrow=mysql_fetch_array($result)){ // 通 过 while 循环 语句 把 这 些 信息 的 内 容 
进行 输出 

14 ?> 

15 <tr align="center" class="stylel"> 

16 <td height="30"><img sre=../<?php echo Smyrow['tb']:?> /></td> 

17 <td><?php echo $myrow['zq'];?></td> 

18 <td><?php echo $myrow['bz']:?></td> 

19 <td><?php echo $myrow['cjsj]:2></td> 

20 <td> 

5 <a hre 全 "delete php?lmbs=<?php echo urlencode(' 栏目 管理 )?>&id=<?php echo 
$myrow['id];?>"> 删 除 </a> 

22 </td> | 

23 </tr> | 

24 。 <?php | 

25 } | 

26 } | 

2 ?> | 

28 </table> | 


最 后 ， 创 建 delete.php 文件 ， 以 超 链 接 传 递 的 值 为 条 件 ， 执 行 delete 语句 删除 指定 的 | 
论坛 专区 数据 。 其 关键 代码 如 下 : | 


01 <2php 
02 session start(); | 
03 include("config.php"); 

04 Sfurl=getenv("http_referer"); 
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05 ifisset($_ SESSION[user]) && isset($_SESSION['pass']){ 
| 06 iflisset($_GET['Imbs"]) && $_GET[Imbs]-=" 用 户 管理 "){ 
| 07 $id-$_ GET['id]: // 获 得 将 被 删除 的 用 户 ID 
| 08 $query="delete from mr_user where id='$id"; // 将 要 删除 的 用 户 从 mr_user 表 中 
| 而 除 
| 09 Sresult=mysql_query($query); 
| 10 if($result) { 
11 echo "删除 成 功 !1"; 
| 12 echo "<meta http-equiv=\"Refresh\" content=\"2;url=$furl\">"; 
| 13 jelsef 
| 14 echo "删除 失败 1"; 
| 号 } 
| 16 } 
| ji iflisset($_GET['Imbs"]) && $_GET["Imbs"]==" 栏 目 管理 "){ 
| 18 $id=$_ GET['id']:; 
| 19 $query="delete from mr _zqfl where id="$id"; 
| 20 Sresult=mysql_query($query); 
| 21 ifSresulb{ 
| 4 echo "<font color=\"#ff0000\"> 删 除 成 功 !1</font>"; 
| 3 echo "<meta http-equiv=\"refresh\" content=\"2; url=$furl\">": 
| 24 } 
| 25 } 
| 26 ifisset($_GET[lmbs]) && $_GET['Imbs']==" 论 坛 主题 "){ 
| 27 $id=$_GET['id]: 
| 28 $query="delete from mr _zqlb where id="$id"; 
| 29 Sresult=mysql_query($query); 
| 30 ifSresulb{ 
| 31 echo "成 功 删 除 !1"; 
| 3 echo "<meta http-equiv=\"refresh\" content=\"2; url=$furl\">"; 
| 33 } 
| 34 } 
| 35 ifisset($_GET[Imbs]) && $_GET[mbs"]==" 回 复 主题 "){ 
| 36 $id=$_GET['id']: 
| 3 Squery="delete from mr_hflb where id="$id™; 
| 38 Sresult=mysql_query($query); 
| 39 if($result){ 
| 40 echo "<font color=\"#ff0000\"> 删 除 成 功 !1</font>"; 
| 41 echo "<meta http-equiv=\"refresh\" content=\"2; url=$furl\">"; 
| 42 } 
| 43 7 
| 44 J}else{ 
| 45 echo "<meta http-equiv=\"Refresh\" content=\"2;url=admin.php\">"; 
| 46 echo "<a hre 伍 \"admin.php\W"> 这 里 </a>"; 
| A 
| a 
”说 明 : 


在 delete.php 文件 中 ， 存 储 了 后 台 管理 系统 中 所 有 删除 操作 的 代码 。 其 根据 超 链接 传 


， 递 的 参 效 值 进行 判断 ， 具 体 执行 哪个 刚 除 语句 ， 


J 
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10.6 技术 提炼 


在 线 论坛 系统 中 用 到 了 分 页 、 网 页 访问 权限 的 设置 和 页 面 刷新 等 技术 。 下 面 对 这 些 技 
术 的 应 用 进行 提炼 、 总 结 。 | 全 ~ 


10.6.1 分 页 技术 
众所周知 ， 网 站 的 分 页 方法 很 多 。 下 面 介绍 笔者 在 本 论坛 中 使 用 的 分 页 技术 是 如 何 实 | 


现 的 


在 线 论坛 中 的 专区 帖子 页 (lmzyphp 文件 ) 中 使 用 了 分 页 技术 ， 实 现 了 输出 页 次 、 页 | 
数 、 共 有 多 少 条 记录 。 此 外 , 在 每 一 页 中 显示 一 条 记录 , 然后 通过 “首页 ””“ 上 一 页 ”“ 下 
一 页 ”以 及 “ 尾 页 ” 超 链 接 来 实现 分 页 的 输出 。 分 页 的 运行 效果 如 图 10.20 所 示 。 
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图 10.20 专区 帖子 


分 页 技术 的 设计 思路 : 首先 使 用 select 查询 语句 ， 从 数据 库 中 查询 出 发 布 的 帖子 的 所 

有 记录 ; 然后 根据 分 页 超 链接 传递 的 分 页 变量 值 ， 以 及 查询 出 的 总 的 记录 数 ， 计 算出 当前 | 
页 输出 数据 的 开始 位 置 ， 接着， 重新 定义 select 查询 语句 ， 应 用 limit 关键 字 控 制 查询 数据 | 
的 开始 位 置 和 查询 的 记录 数 ;， 最 后 ， 通 过 while 语句 循环 输出 查询 结果 。 其 具体 操作 代码 | 
如 下 (lmzy.php 文件 ): | 
01 <?php | 

02 include("config php"): | 

03 ”S$page=(isset($_GET['page']))?$_GET['page'"]:"1": /获取 分 页 超 链接 传递 的 分 页 变量 值 | 

04 S$zq=(isset($ GET['zq']))?$ GET['zq']:"PHP"; | 


05 > 
06 <?php 
07 。 $page_size=1: // 每 页 显示 1 条 记录 


08 $query="select * from mr zqlb where zq=".$zq.";// 从 数据 库 中 查询 所 有 符合 条 件 的 数据 
09 Sresult=mysql query(Squery); 


10 $message count=mysql num rows(S$result); // 统 计 总 的 记录 数 

11 if($message_count>0){ 

12 > 

13 <table width="776" height="30" border="1" align="center" cellpadding="1" cellspacing="1"> 
14 <tr align="center"> 

15 <td width="51" height="26" class="style1"> 状 态 </td> 


.953 本 


16 <td width="54" class="stylel"> 心 情 </td> 


17 <td width="381" class="stylel"> 主 &nbsp:&nbsp:&nbsp:&nbsp:&nbsp: 题 </td> 

18 <td width="79" class="style1"> 作 者 </td> 

19 <td width="76" class="style1"> 回 复 /人 气 </td> 

20 <td width="135" class="stylel"> 发 表 时 间 </td> 

Zh </tr> 

22 ”<?zphp 

23 Spage_count=ceil($message_count/$page_size); 1/ 计算 总 的 页 码 数 

24 S$offset=($page-1)*$page size; /计算 当前 页 查询 的 开始 位 置 
25 $query="select * from mr _zqlb where zq="".$zq." order by id desc limit $offset ,$page size "; 
26 $result=mysql query($query): // 查 询 当 前 页 显示 的 记录 

27 while($Smyrow=mysql fetch_array(Sresulb){ /循环 输出 查询 结果 

28 ?> 

29 <!-- 省 略 部 分 代码 --> 

30 <?php 

a 

2 ?> 


33 </table> 


上 述 代 码 完 成 了 分 页 查询 操作 ， 以 及 查询 结果 的 循环 输出 。 下 面 介绍 分 页 超 链接 的 创 
建 ， 其 关键 代码 如 下 : 


01 <table width="100%" border="0" cellspacing="0" cellpadding="0"> 
02 <tr class="stylel"> 
03 <td width="40%"> 页 次 <?php echo $page:?>/<?php echo $page_count;?> 页 记录 : <?php echo 


”$message_count:?> 条 </td> 


”Spaget+1;?>"> 下 一 页 </a> 


04 <td width="50%"> 分 页 : 

05 <a href='Imzy.php?zq=<?php echo urlencode($zq);?>&page=1> 首 页 </a> 

06 <?php 

07 if($page >= 2){ 

08 ?> 

09 <a href="Imzy.php?zq=<?php echo urlencode($zq):?>&page=<?php echo 
Spage-1;?>"> 上 一 页 </a> 

10 <?php 

11 } 

12 if($page < $page_count){ 

13 ?> 

14 <a href="lmzy.php?zq=<?php echo urlencode($zq);?>&page=<?php echo 

15 <?php 

16 } 

17 if($page <= $page_count){ 

18 ?> 

19 <a href="lmzy.php?zq=<?php echo urlencode($zq);?>&page=<?php echo 
$page_count;?>"> 尾 页 </a> 

20 <?php 

Zl } 

2 > 

23 </td> 

24 </tr> 
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25 </table> 


10.6.2 ”网 页 访问 权限 设置 


网 页 访问 权限 设置 的 目的 是 控制 不 同 用 户 拥有 不 同 的 权限 。 例如 , 在 后 台 管理 系统 中 ，| 力 f 
只 有 管理 员 才 具 备 访问 权限 , 可 以 执行 各 种 操作 , 其 他 用 户 不 能 访问 后 台 管理 系统 的 网 页 ， | 
而 在 论坛 系统 中 ， 只 有 登录 成 功 的 会 员 才 具备 发 布 和 回复 帖子 的 权限 ， 普 通 游客 不 具备 此 ! 
权限 。 那 么 这 个 网 页 访问 的 权限 是 如 何 设置 的 呢 ? | 

这 里 以 后 台 管 理 系统 中 的 网 页 为 例 ， 讲 解 如 何 设置 网 页 访问 权限 。 其 关键 是 会 话 | 
(SESSION) 的 应 用 ， 它 解决 了 HTTP Web 协议 的 无 状态 问题 ， 实 现 数据 在 不 同 页 面 之 间 | 
的 传递 。 在 此 设置 网 页 的 访问 权限 就 是 通过 判断 SESSION 变量 的 值 来 完成 的 。 

首先 ， 在 后 台 管理 员 登 录 成 功 后 ， 将 管理 员 的 用 户 名 和 密码 存储 到 SESSION 变量 中 。 
然后 ， 在 后 台 管 理 系统 的 网 页 中 ， 通 过 $_ SESSION 获取 SESSION 变量 中 存储 的 用 户 名 和 | 
密码 ， 通 过 isset() 函 数 判断 其 是 否 存在 ， 如 果 存在 则 说 明 当前 用 户 可 以 访问 此 网 页 ， 否则 | 
当前 用 户 就 不 可 以 访问 此 网 页 ， 给 出 提示 信息 ， 跳 转 到 后 台 管理 系统 的 登录 页 。 其 关键 代 | 
码 如 下 : | 


01 <?php 

02 session_start(); 1/ 初始 化 SESSION 变量 
03 iftisset($_SESSION["user']) && isset($_SESSION['pass])){ /判断 当前 用 户 的 权限 
04 ?> 

05 <!-- 省 略 了 部 门 代码 --> 

06 <?php 

07 }else{ 

08 echo "<meta http-equiv=\"Refresh\" content=\"2;url=admin.php\">"; 

09 echo "<a hre 人 =\"admin.php\"> 这 里 </a>"; 

ms 

11 > 


10.6.3 页面 刷新 技术 


为 了 能 够 更 好 地 浏览 到 论坛 中 的 信息 ， 笔 者 在 论坛 中 采用 了 页 面 刷新 技术 ， 以 此 来 保 | 
持 页 面 中 始终 输出 最 新 的 数据 。 此 技术 的 实现 应 用 的 是 一 个 简单 的 JavaScript 语句 ， 其 代 
人 码 如 下 : 


<a hre 全 "javascript:location_ reloadO" target="_self"> 刷 新 页 面 </a> 
通过 单 击 “ 刷 新 页 面 ” 超 链 接 ， 即 可 完成 对 页 面 数据 的 更 新 操作 。 | 
此 外 ， 还 有 一 种 刷新 页 面 的 方法 ， 可 以 控制 刷新 的 时 间 ， 即 每 隔 一 段 时 间 对 页 面 刷新 | 
一 次 。 其 代码 如 下 : | 
<meta http-equiv="refresh" content="2;url=index.php"> 
上 述 代 码 的 含义 是 每 阳 2 秒 钟 对 index.php 页 刷新 一 次 。 
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( 镶 d 自学 视频 、 源 程序 : 配套 资源 \mmN11\ ) 


如 果 有 人 问 :“ 什 么 是 Web 2.0? ”， 丽 怕 没 几 个 人 能 说 得 清楚 ， 但 是 要 问 
“什么 是 博客 ? "“， 哪 怕 是 从 不 上 网 的 人 也 耳熟能详 。 博 客 正 是 Web 2.0 概念 
中 重要 的 组 成 部 分 之 一 (其 他 众所周知 的 还 包括 1M 即时 通 和 R55 阅读 器 )。 

Blog (博客 )， 全 名 Weblog, 后 来 缩写 为 Blog。Blogger 就 是 写 Blog 的 人 ， 
习惯 于 在 网 上 写 日 记 、 发 布 个 人 照片 、 展 示 个 性 自我 的 用 户 群体 。 对 于 
Blog/Blogger 的 中 文 名 称 ， 有 翻译 成 “博客 ”的 ， 也 有 翻译 为 “网 志 ” 的 ， 但 
大 多 数 人 都 已 经 认可 了 “博客 "。 通 过 阅读 本 章 ， 读 者 可 以 学 到 以 下 内 容 。 


WI 系统 设计 思路 

MW 数据 库 设计 

MW 博客 空间 个 人 首页 设计 
WI 我 的 文章 模块 的 设计 
WI 文章 浏览 模块 设计 
文章 管理 模块 设计 

好 友 管 理 模块 设计 

小 纸 条 管理 模块 设计 
用 户 管理 模块 设计 
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11.1 博客 概述 


11.1.1 ”模块 概述 全 中 


博客 ， 即 在 网 络 上 发 表 文章 、 展 现 个 性 的 一 类 人 ， 而 博客 系统 ， 是 为 这 类 人 群 提供 一 个 
相互 交流 、 学 习 的 平台 。 博 客 的 两 大 基本 功能 是 共享 与 交流 。 共 享 ， 就 是 将 文章 、 图 片 、 心 | 
得 等 一 些 在 传统 概念 中 属于 私人 的 东西 ， 拿 出 来 和 多 数 人 一 起 分 享 ， 交 流 ， 就 是 有 着 同样 兴 | 
趣 、 爱 好 、 语 言 的 一 类 人 之 间 的 联系 。 


11.1.2 ”功能 结构 


本 章 的 博客 系统 主要 分 为 个 人 博客 管理 和 博客 后 台 管理 两 部 分 。 | 
个 人 博客 管理 ， 主 要 功能 包括 个 人 管理 、 文 章 管理 、 相 册 管 理 、 好 友 管 理 、 留 言 | 
管理 和 小 纸 条 管理 。 个 人 博客 管理 功能 结构 如 图 11.1 所 示 。 | 


个 人 博客 管理 


[好 赤 管理 | [BE |] [4 条 | 


图 11.1 个 人 博客 管理 功能 结构 图 
回 ”博客 后 台 管 理 : 主要 是 对 博客 用 户 和 管理 员 的 管理 。 其 主要 功能 包括 用 户 管理 、 


用 户 文章 管理 、 用 户 相册 管理 、 管 理 员 管理 、 系 统 日 志 管理 和 数据 备份 、 恢 复 。 
。257 。 四 
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个 人 博客 管理 页 面 运行 效果 如 图 11.4 所 示 。 


图 11.4 个 人 博客 管理 页 面 
博客 用 户 管理 页 面 运行 效果 如 图 11.5 所 示 。 | 
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图 11.5 博客 用 户 管理 页 面 运行 效果 | 
图 片 管理 页 面 运行 效果 如 图 11.6 所 示 。 | 
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图 11.6 图 片 管理 页 面 运行 效果 
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a VAI 
11.2 数据 库 设 计 


大 本 模块 属于 中 小 型 个 人 网 站 ， 毫 无 争议 地 ,依然 采用 PHP+MySQL 这 对 黄金 组 合 (天 
名 | 论 是 从 成 本 、 性 能 、 安 全 ， 还 是 从 易 操作 性 上 考虑 ，MySQL 都 是 最 佳 选择 )。 
SJ3 玫 11.2.1 创建 数据 库 与 数据 表 


| “博客 系统 使 用 的 数据 库 是 db_blog, 其 中 包含 9 张 数据 表 , 各 表 的 名 称 和 作用 如 图 11.7 
| 所 示 。 


晤 
EE 
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图 11.7 数据 表 列 表 
下 面 来 具体 看 一 下 其 中 几 张 重要 数据 表 的 结构 设计 。 
1. tb_article (博客 文章 列表 ) 


| tb_article 表 中 包含 文章 的 类 别 、 文 章 标题 、 文 章 内 容 等 信息 ， 其 结构 及 字段 说 明 如 
| 图 11.8 所 示 。 


CGIZTOEETEITOEEETCTD 
上 全 作 。 天 清 裤 ”下 


图 11.8 tb_article 表 结构 
2. tb frd (用 户 好 友信 息 表 ) 


| tb_frd 表 中 包含 好 友 名 称 、 添 加 人 、 好 友 等 级 和 添加 时 间 等 字段 ， 其 结构 及 字段 说 
| 明 如 图 11.9 所 示 。 


ame unestanp vr cm nese 再。 CURRENT TIMESTAMP El 


图 11.9 tb_frd 表 结构 
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3. tb_member ( 用户 信息 数据 表 ) 


tb_member 表 中 包含 用 户 的 基本 资料 ， 如 E-mail、 姓 名 、 出 生日 期 等 ， 以 及 个 人 博 
客 中 的 一 些 参数 ， 如 点 击 率 、 文 章 类 别 、 相 册 类 别 、 是 否 推荐 等 。tb_member 表 结构 及 ; 
说 明 如 图 11.10 所 示 。 | 鲜 | 
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图 11.10 tb_member 表 结 构 
4. tb_review 表 (评论 列表 ) 


tb_review 表 中 包含 被 评论 的 文章 id、 发 表 评论 人 、 评 论 内 容 以 及 评论 时 间 ， 其 结构 如 
图 11.11 所 示 。 


隔 珊 和 locaihoer ， 司 委 册 亩 db-biog 日 委 “threview 
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图 11.11 tb_review 表 结构 | 
5. tb uppics 表 (上传 图 片 列表 ) | 


tb_uppics 表 中 包含 上 传 图 片 的 信息 ， 如 上 传 图 片 名 称 、 上 传 用 户 、 上 传 类 别 等 ， 其 结 
构 如 图 11.12 所 示 。 


[ET ETITITDEEETTTTS 


图 11.12 tb_uppics 表 结构 


2 


多 (3 汪 ysQrI 扩展 在 以 下 方面 有 了 明显 的 提高 。 


| 幅 ， 这 是 


Op 闫 衣柜 决 精 角 


此 外 ， 还 有 管理 员 信息 表 、 文 章 日 志 信 息 表 、 用 户 留言 数据 表 、 小 纸 条 列表 ， 限 于 篇 


| 三 22 


不 再 介绍 ， 读 者 可 参见 配套 资源 中 的 数据 库 文件 。 
数据 库 操作 类 


为 了 方便 使 用 、 减少 代码 ， 本 模块 采用 MySQLI 扩展 存 取 数据 。 与 MySQL 扩展 相 比 ， 


回 


[al 


兼容 性 与 维护 性 : MySQLI 拥有 比 MySQL 更 高 的 兼容 性 ，MySQLI 扩展 可 以 很 
容易 地 使 用 MySQL 的 新 功能 ,即使 MySQL 的 新 版 本 又 出 现 了 更 多 功能 ,MySQLI 
扩展 也 可 以 很 容易 地 支持 。 

面向 对 象 ; MySQLI 扩展 已 封装 到 一 个 类 中 ， 从 而 可 使 用 面向 对 象 的 方式 编程 。 
即使 对 面向 对 象 不 了 解 ，MySQLI 也 可 以 利用 面向 过 程 的 编程 方式 来 实现 。 
速度 和 安全 性 : MySQLI 扩展 执行 的 速度 要 比 之 前 版 本 的 MySQL 扩展 快 了 很 多 。 
MySQLI 扩 展 支持 MySQL 新 版 本 的 密码 杂凑 (Password Hashes) 和 验证 程序 ， 
大 大 提高 了 应 用 程序 的 安全 性 。 

预 准备 语句 ， 预 准备 语句 可 提高 重复 使 用 的 语句 的 性 能 ，MySQLI 扩展 提供 了 对 
预 准备 语句 的 支持 。 

调试 功能 : MySQLI 扩展 进一步 改进 了 调试 功能 ， 提 高 了 开发 效率 。 要 在 PHP 中 
使 用 MySQLI 扩展， 需要 在 配置 文件 php.ini 中 添加 “extension=php_mysqli.dll” 
设置 ， 如 果 配 置 文件 中 己 有 上 述 设置 ， 则 应 确保 extension 的 前 面 没有 “;”， 否 则 
将 其 去 掉 。 


下 面 介绍 如 何 使 用 MySQLI 扩展 来 存 取 数 据 库 。 MySQLI 提供 了 面向 对 象 和 面向 过 程 


两 种 方式 来 与 数据 库 交 互 。 在 本 模块 中 ， 后 台 管理 应 用 面向 对 象 方式 与 数据 库 交 互 ， 前 台 
”应 用 面向 过 程 方式 与 数据 库 交互 。 


下 面 是 后 台 管 理 MySQLI 扩展 应 用 的 面向 对 象 操作 类 。 在 这 里 连接 数据 库 就 是 实例 化 


(1 


MySQLI 系统 内 定 的 一 个 类 ， 将 所 得 到 的 返回 集 “ $this->conn ”以 “ Sthis->result = 
| $this->conn->query($sql)” 的 形式 写 入 。 其 存储 于 admin\conn\conn.php 文件 中 。 步 又 如 下 : 


) 定义 私有 属性 ， 如 主机 名 、 用 户 名 、 用 户 密码 、 数 据 库 名 等 。 
01 <?php 
02 class opmysqli{ 
03 private $host ="127.0.0.1'; 1/ 服务器 地 址 
04 Private $name = 'root'; // 登 录 账 号 
05 private Spwd = "111'; // 登 录 密码 
06 private $dBase = 'db_blog': /数据 库 名 称 
07 private $conn ="; /| 数据 库 连接 资源 
08 private $result ="; // 结 果 集 
09 private $msg = "; // 返 回 结果 
10 private $fields; // 返 回 字 段 
11 Private $fieldsNum = 0; // 返 回 字 段 数 
| bp Private $rowsNum = 0; 1/ 返回 结果 数 
13 Private $filesArray = array(); /返回 字段 数组 
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14 Private $rowsArray = array(O; // 返 回 结果 数 组 | 

(2) 使 用 _construct 关键 字 初 始 化 类 。 这 与 MySQL 扩展 写法 一 样 。 | 
01 function _ construct($host=",$name=",$pwd=",$dBase="){ // 初 始 化 类 | pp 
02 if($host !=") | 和食 斤 | 
03 Sthis->host = $host; | 和 
04 这 $name (=") ‘Note 
05 Sthis->name = $name; ! 
06 if(Spwd (=") | 
07 Sthis->pwd = $pwd; | 
08 if($dBase !=") | 
09 Sthis->dBase = $dBase: | 
10 S$this->init_conn(); | 
11 | 


(3) 连接 数据 库 需 要 实例 化 MySQLI 类 。 这 个 类 是 PHP 环境 自 带 的 ， 所 以 不 要 管 它 
是 从 哪里 来 的 ， 只 需 实例 化 即 可 。 


01 function init_conn|O{ 


02 $this->conn=new mysqli($this->host.,$this->name,$this->pwd,$this->dBase); 
03 $this->conn->query("set names gb2312"): 
04 1 


(4) 下 面 是 一 些 具体 的 操作 方法 ， 包 括 查询 结果 、 取 得 字段 数 、 更 新 、 删 除 、 添 加 
记录 数 等 。 其 部 分 代码 如 下 : 


01 /查询 结果 

02 function mysqli query_rst($sqD){ 

03 if($this->conn =— "){ 

04 Sthis->init_conn(); 

05 } 

06 Sthis->result = (@$this->conn->query($sq]); 

07 } 

08 /取得 字段 数 

09 function getFieldsNum($sqD){ 

10 $this->mysqli query_rst($sq]l):; 

11 S$this->fieldsNum = (@$this->result->num fields($this->result): | 
12 ) | 
13 // 取 得 查询 结果 数 | 
14 function getRowsNum($sqD){ | 
15 Sthis->mysqli query_rst($sql): | 
16 Sthis->rowsNum = (@$this->result->num rows; | 
] return $this->rowsNum:; | 
18 } | 
19 // 取 得 结果 数值 | 
20 function getRowsArray($sql){ | 
21 Sthis->mysqli_ query_rst($sqD: | 
22 while($row = (@S$this->result->fetch_array(MYSQLI ASSOC)) { | 
23 Sthis->rowsArray[] = $row:; | 
24 } 
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»5 return $this->rowsArray: 
26 

27 /更 新 、 删 除 、 添 加 记录 数 
28 function uidRst(SsqD{ 

29 if($this->conn — "){ 

30 Sthis->init_conn(); 
31 } 

32 S$this->conn->query($sq]); 
33 S$this->rowsNum = (@S$this->conn->affected_rows; 
34 return $this->rowsNum:; 
35 } 


(5) 最 后 实例 化 opmysqli 类 即 可 。 
01 $conne = new opmysqliO; 
02 ?> 
前 台 使 用 MySQLI 以 面向 过 程 方式 与 数据 库 进 行 交 互 。 其 使 用 方法 类 似 于 MySQL 扩 

| 展 。 例 如 ，mysql_query 这 个 MySQL 扩展 函数 ， 在 MySQLI 扩展 中 写成 mysqli_query。 返 
| 回 结果 集 在 MySQL 中 是 mysql query($sql ，S$conn) ， 而 在 MySQLI 扩展 中 则 是 
| mysqli_query($conn,$sql)。 下 面 给 出 MySQLI 面向 过 程 操 作 数 据 库 的 部 分 代码 ， 其 存储 于 
center/conn/conn.php 文件 中 。 


01 ” /连接 数据 库 

02 function init_connO{ 

03 Sthis->conn=mysqli_connect($this->host,$this->name,$this->pwd,$this->dBase); 
04 mysqli_query(S$this->conn,"set names gb2312"); 

05 } 

06 ”// 查 询 结果 

07 function mysqli query_rst($sqD){ 

08 if($this->conn =— "){ 

09 Sthis->init_conn(); 

10 } 

11 Sthis->result = (Dmysqli_query($this->conn,$sq]); 

12 Hi 

13 // 取 得 字段 数 

14 function getFieldsNum($sqD){ 

15 Sthis->mysqli_query_rst($sq]l): 

16 Sthis->fieldsNum = @mysqli num fields($this->result); 
he } 


至 此 ， 有 关 数 据 库 操作 类 的 讲解 完毕 ， 其 具体 代码 参见 配套 资源 中 内 容 。 


| 11.3 博客 室 间 个 人 首页 


11.3.1 博客 空间 个 人 首页 概述 
博客 系统 的 模块 较 多 ， 限 于 篇 幅 ， 这 里 只 对 重点 模块 进行 讲解 。 首 先 来 看 个 人 首页 。 
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个 人 首页 由 多 个 页 面 组 成 ， 如 热门 文章 、 文 章 分 类 、 留 言 列表 等 。 其 页 面 的 设计 结构 如 
图 11.13 所 示 。 | 


EREETETES 
这 | 便 - 
加 未 新 汕 息 ”| 导航 栏 ( 我 的 文章 我 的 相 朋 我 的 资料 我 的 好 友 ) | 己 ~ 
热门 文章 列表 | 
博客 信息 
Cprmouinlo pip) Chota Note 
' 
| 
文章 类别 风衣 | 
Carttype ) ! 
日 历 -9 | 
Ca linl) 内 容 显示 区 | 
(Minewat! php) | 
| 
图 片 类 别 列表 | 
Cpictype php) | 
| 
留言 列表 | 
【 1 ) | 
bmg 好 友 泛 按 | 
《anfrd php) ! 
| 
留言 板 ( yb php) | 
| 


! 

图 11.13 博客 空间 个 人 首页 结构 及 布局 | 

上 

个 人 首页 的 运行 效果 如 图 11.14 所 示 。 | 


图 11.14 个 人 首页 运行 效果 | 
11.3.2 ”首页 实现 过 程 | 


个 人 首页 设计 非常 简单 ， 总 体 上 采用 DIV+CSS 来 实现 (如 导航 条 、 个 人 首页 的 头 部 ，| 
还 有 外 层 等 )， 然 后 运用 包含 文件 结合 技术 ， 把 所 有 需要 显示 在 首页 的 文件 和 配置 文件 引 | 
入 进来 就 可 以 了 。 其 具体 步 又 如 下 : | 
(1) 在 center\center.php 文件 中 ， 加 载 各 种 文件 ， 如 配置 文件 、 数 据 库 类 文件 、 计 数 | 
统计 文件 、CSS 样式 文件 、JS 脚本 文件 等 。 代 码 如 下 : | 


01 <Iphp 
02 session start(); /开启 session 
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03 header('Content-Type:text/html:;charset=gb2312"); // 设 置 页面 编 码 
04 include once '../config php'; // 加 载 配 置 文件 
05 include_once 'conn/conn php'; /加 载 数据 库 类 
06 include_once 'inc/count.php'; /加 载 计数 器 
07 ?> 


08 <link href="../css/style.css" rel="stylesheet" type="text/css" /> 

09 <script language="javascript" sre="../js/xmlhttprequest.js"></script> 
10 <script language="javascript" sre: s/center.js"></script> 

11 <script language="javascript" sre: s/date.js"></script> 

12 <script language="javascript" sre="../js/personinfo.js"></script> 


(2) 创建 “登录 ” “注册” 超 链接 。 判 断 当前 的 SESSION 值 是 否 为 空 ， 如 果 为 空 ， 就 
显示 “登录 ”“ 注 册 ” 超 链接 ; 否则 ， 显 示 光 临 信息 和 进入 后 台 管 理 的 超 链 接 。 代 码 如 下 : 


01 <?php 

02 if(!empty($_SESSION['name']) and !is_null($_ SESSION[mame'])){ 
03 ?> 

04 您 好 ， 欢 迎 光临 <?php echo $_SESSION[mame']; ?> 的 博客 ! 


05 <aid-blogmain' onclick="javascript:open('manage.php', parent,",false)"> 进 入 管理 页 面 </a>&nbsp; 
06 <aonclick="javascript:open(logout.php', parent,",false)"> 退 出 登录 </a>&nbsp; 


07 <2php 

08 }else{ 

09 ?> 

10 <a onclick="javascript:window.open('login.php','login','width=300,height=200',false)"> 登 
录 </a> 

11 <a onclick="javascript:window.open('register.php','register',",false)"> 注 册 </a>&nbsp; 

12 <2php 

13 

14 ?> 


(3) 创建 Banner， 并 且 输 出 博客 空间 的 名 称 。Banner 下 面 的 一 行 由 两 部 分 构成 ， 一 
部 分 是 新 消息 提醒 ， 另 一 部 分 是 导航 栏 。 新 消息 提醒 每 隔 一 段 时 间 ， 查 看 一 下 个 人 的 纸 条 信 
息 ， 如 果 发 现 新 纸 条 ， 会 马上 显示 出 来 。 代 码 如 下 : 


01 <div id="nav"> 


02 <div id="newscrip">&nbsp: 

03 <script>setInterval("shownewmess()",10000);</script> 
04 </div> 

05 <div id="rightfloat"> 


06 <a onclick='javascript:open("center.php?uid=<?php echo $ GET[uid] ?>"," parent 
","",false) 人 > 我 的 文章 </a> 

07 | <a onclick='javascript:open("center.php?act=pics&uid=<?php echo $ GET[uid] ?> "," parent"," 
",false) 人 > 我 的 相册 </a> 

08 | <a onclick='javascript:open("center.php?act=person&uid=<?php echo $ GET[mid] ?> ","_parent"," 
"false)> 我 的 资料 </a> 

09 | <a onclick=javascript:open("center php?act=frd&uid=<?php echo $ GET[uid] ?>"," parent","".false)> 我 
的 好 友 </a></div> 

10 </div> 
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(4) 加 载 各 个 页 面 。 其 中 ， 内 容 显 示 区 是 首页 的 核心 ， 几 乎 所 有 博客 空间 的 功能 ， 


都 将 在 内 容 显示 区 显示 。 内 容 显示 区 的 代码 如 下 : 


01 oo | 

02 <2php | 贪 内 
03 ifisset($_GET[act]) and $_ GET['act] 一 "){ | YY 
04 include_once "newartl.php': Note 
05 }else iflisset($_GET['act]) and $ GET['act] =— 'see){ 8 

06 include_once 'showarticlel .php'; | 

07 }else iflisset($_GET[act]) and $ GET['act] == 'pics){ | 

08 include_once 'picsl .php'; | 

09 }else iflisset($_GET['act]) and $_ GET['act] 一 'person"){ | 

10 include_once 'person.php': | 

到 jelse iflisset($_GET['act]) and$_GET['act] 一 'frd){ | 

12 include_once 'frd1.php'; | 

13 }else iflisset($_GET['act]) and $_GET['act] == "modart){ | 

14 include_once 'modartphp': | 

15 jelse{f | 

16 include_once 'newartl .php’; | 

17 } | 

18 ?> 

19 </div> 

20 ss 


11.4 我 的 文章 


11.4.1 “我 的 文章 ” 概 


进入 博客 个 人 空间 ， 或 者 单 击 博客 个 人 空间 中 的 “我 的 文章 ” 超 链接 ， 将 在 内 容 显 示 
区 显示 博客 中 已 存在 的 文章 的 简要 信息 列表 ， 其 中 包括 文章 标题 、 发 布 时 间 、 部 分 文章 内 | 
容 和 一 个 “阅读 全 文 ” 超 链接 ， 如 图 11.15 所 示 。 | 


文章 标题 : 公 
Re 加 接 中 国 数 字 出 版 的 腾飞 。? 数 字 化 出 版 ,改变 你 我 生 
活 。? 让 用 户 花 更 少 的 钱 ， 获 得 更 多 、 更 专业 、 更 有 价值 的 内 容 和 服务 。? 为 用 户 


| 
| 
! 
| 
! 
所 文章 列表 | 
| 
| 
! 
| 
创造 价值 ， 就 是 为 自己 创造 品牌 。? 让 忧 秀成 为 习惯 。? 坚 韦 、 创 新 、…… ! 


文章 标题: 公司 要 况 发 
Patent 明 | ee 、 司 是 _ 家 届 计 算 机 软件 和 数字 出 版 核心 的 训 科 技 企业 ， | 

域 的 实践 ， 目前 已 和 多 家 国 ! 
EE 公司 的 多 款 软 件 产品 应 用 于 国内 的 多 ~ ! 


| 现款 全 文 | 


图 11.15 文章 列表 的 运行 效果 
单 击 “ 阅 读 全 文 ” 超 链接 :打开 showarticlel.php 页 面 ,将 显示 要 查看 的 文章 的 全 文 和 
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评论 信息 。 同 时 ， 还 会 显示 相关 的 一 些 操作 ， 例 如 引用 文章 和 发 表 评 论 。 如 果 是 本 人 的 博 
| 客 ， 登 录 后 还 可 以 修改 文章 、 删 除 文章 和 删除 评论 ， 如 图 11.16 所 示 。 


| 图 11.16 浏览 文章 页 面 的 运行 效果 
”11.4.2 “我 的 文章 ”功能 实现 过 程 


| 创建 newartl php 文件 ， 完 成 “我 的 文章 ”模块 设计 。 首 先 设置 分 页 信息 ， 然 后 生成 
| SQL 查询 语句 ， 用 来 查找 当前 用 户 所 发 表 过 的 文章 ， 并 返回 查询 结果 。 实 现代 码 如 下 : 


01 <?php 
02 include_once 'conn/conn.php'; 
03 ifisset($_GET[uid])){ 
04 Snum = 7; // 每 页 显示 记录 数 
05 if(!lisset($_GET['curpage']) ){ 
06 Scurpage = 1; 
07 jelse{ 
08 S$curpage = $_GET['curpage']; // 当 前 页 
09 } 
10 $sql = "select * from tb_article where author = ".$_GET['uid']." order by id desc "; 
11 S$totnum = $conne->getRowsNum($sq]): // 记 录 数 
12 S$totpage = ceil(Stotnum / Snum):; // 页 数 
13 Stmpsql = $sql." limit ".($num *($curpage-1)).",".$num; //SQL 查询 语句 
| 14 $arr = $conne->getRowsArray($tmpsq]); 1/ 获取 数组 
| 15 Sconne->close_rst(; 
| TCR 


如 果 获 取 到 的 数组 不 为 空 ， 那 么 循环 判断 是 否 为 引用 ， 如 果 不 是 ， 则 直接 输出 文章 信 
| 息 ， 如 果 是 ， 那 么 根据 引用 的 文章 ID 号 来 查找 原文 ， 并 输出 文章 信息 。 最 后 ， 添 加 一 个 
| “阅读 全 文 ” 超 链接 。 代 码 如 下 : 

01 if(gar!="){ 
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02 foreach($arr as $value){ | 

03 这 !empty($value['artquote']) and !is_null($value['artquote']){ | 

04 $tmparr = explode(',', $value['artquote']); | 

05 $sqll = "select * from tb_article where id = ".$tmparr[0]; | 

06 Sartarrl = $conne->getRowsArray($sql]); | 贪 内 
07 $conne->close_ rstO; | RN 
08 ws 

09 }else{ | yote 
10 3 | 

11 } | 

12 } 


11.4.3 ”文章 浏览 功能 实现 过 程 | 


在 文章 浏览 页 (showarticlel.php) 中 ， 最 核心 的 功能 当然 是 浏览 文章 。 首 先 ， 根 据 超 
链接 传递 的 文章 ID， 将 文章 信息 全 部 显示 出 来 。 代 码 如 下 : 


01 。 <?php | 
02 include_once '../config.php'; // 载 入 配置 文件 | 
03 include_once 'conn/conn.php'; // 载 入 数据 库 类 | 
04 $showartsql] = "select * from tb_article where id = ".$_GET['artid']; /生成 查询 语句 | 
05 $artarr = $conne->getRowsArray($showartsq]); // 返 回 数组 形式 的 记录 | 
06 $conne->close_rst(); // 清 空 结 果 集 | 
07 if(!empty($artarr[0]['artquote"]) and !is_null($artarr[0]['artquote'])){ // 判 断 文 章 是 | 
否 为 引用 | 
08 Stmparr = explode(",,, $artarr[O]['artquote']): // 得 到 引用 的 来 源 地 址 | 
09 $sqll = "select * from tb_article where id =".$tmpar[0];// 根 据 地址 查询 数据 | 
10 $artarrl = $conne->getRowsArray($sql1): // 返 回 结 果 集 | 
11 $conne->close rstO; | 
12 交 | 
13 Jelse{ | 
14 2 | 
15 } | 
I | 


然后 ， 实 现 文章 引用 《〈 其 实质 是 引用 文章 地 址 )。 当 其 他 的 博客 用 户 看 到 好 文章 时 ， | 
可 以 通过 “引用 ” 超 链接 ， 将 文章 ID 保存 到 自己 的 文章 列表 中 。 通 过 对 文章 ID 的 访问 ， | 
可 以 省 去 手工 输入 的 麻烦 ， 也 在 一 定 程度 上 节省 了 表 空 间 。 此 外 ， 如 果 博 主 想 对 自己 的 这 
篇 文章 进行 宣传 ， 也 可 以 将 引用 地 址 复制 下 来 ， 粘 贴 到 其 他 地 方 来 提高 博客 空间 的 人 气 。 
实现 引用 的 代码 如 下 : 
[<a onclick="retum artquote('<?php echo $ GET[mid] ?>'"<?php echo $ GET['artid]; ?>")" 
style="color: 姓 F6600:"> 引 用 </a>]&nbsp: 来 自动 添加 到 您 的 博客 空间 <br 记 | 
<input id="copyadd" onclick="copyaddress(" type="text" size="50" value="<?php echo | 
‘http://.$_SERVER['HTTP_ HOST'].$ SERVER['REQUEST URI]: ?>" readonly="readonly" style="border: 
lpx #FFFFFF solid:"” /> | 


该 段 代码 主要 通过 两 个 JavaScript 自 定义 函数 来 控制 。 第 一 个 函数 artquote() 将 博客 用 
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户 、 文 章 ID 传 给 showarticle_chk.php 页， 根据 返回 结果 进行 处 理 。 脚 本 文件 js\centerjs 的 
| 代码 如 下 : 


01 ”1// 引 用 
02 function artquote(uid,quoteid){ 
03 xmlhttp.open('get','showarticle_chk.php?act=quote&uid='+uid+'&quoteid="+quoteid,true); 
04 xmlhttp.onreadystatechange = functionO{ 
05 if(xmlhttp.readystate == 4 && xmlhttp.status 一 200){ 
| 06 msg = xmlhttp.responseText; 
| 07 iftmsg — 1){ 
| 08 alert( 引 用 成 功 '); 
| 09 jelseifmsg — 2){ 
| 10 alert(' 您 还 没有 登录 ); 
| 加 }else if(msg =— '3"){ 
| 2 alert( 引 用 失败 "); 
| 13 }else if(msg — '4){ 
| 14 alert( 不 允许 引用 自己 的 文章 ); 
| 15 }else{ 
| 16 alert(msg): 
| 17 } 
| 1 } 
19 
20 xmlhttp.send(nul]); 
2 


showarticle_chk.php 页 处 理 和 文章 相关 的 操作 ， 包 括 添 加 评论 、 删 除 评论 和 引用 。 其 
主要 代码 如 下 : 


01 <?php 
02 session start():; /开启 SESSION 
03 header('Content-Type:text/html;charset=gb2312"); // 设 置 编 码 
04 include_once 'conn/conn.php'; // 载 入 数据 库 类 
05 $artid = $_GET['artid"]: /文章 ID 
06 S$quoteid =$_ GET['quoteid']; /引用 了 
07 $name =$_SESSION[mame']; /当前 用 户 
08 $uid=$_GET[muid]; // 原 文章 用 户 
| 09 S$act=$_GET['act]: // 何 种 操作 
| 10 Sreback = '0'; 
| 11 ifempty(Sacbj{ 
| 12 if($name != Suid){ 
| 13 Sselectsql] = "select renum from tb_article where id = ".$artid; 
| 14 Srenum = $conne->getFields($selectsql.0): 
| 15 $upsql = "update tb_article set renum = ".(++$renum)." where id = ".$artid; 
| 16 Snum = $conne->uidRst($upsq]); 
17 中 
| 18 S$content =$_GET[cont]:; 
| 19 $sql = "insert into tb_review (artid,content,man) values("".$artid.",".$content.",".($name 一 
"9" 匿名 ':$name).")"; 
| 20 Snum = $conne->uidRst($sq]): 
| 21 Sreback = "1'; 
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2 jelse if($act 一 'quote){ /如 果 是 引用 操作 
23 这 !empty($name) and !is_null($name)){ 
24 ifS_GET[mid] 一 Sname){ /如 果 是 引用 了 自己 的 文章 
25 Sreback = '4'; 
26 }else{ 
27 $chkquote = $conne->getRowsNum("select artquote from tb article where 
author = ".$name."™™"); 
28 if($chkquote >= 1){ a 
29 Sreback = "5"; | 
30 jelsef | 
31 Sselectsql = "select renum from tb_article where id =".$quoteid; Z| 
32 Srenum = $conne->getFields($selectsql.0); | 
33 $upsql = "update tb_article set renum = ".++$renum." where id = | 
".$quoteid; | 
34 Snum = $conne->uidRst($upsq]); | 
35 $oldid = $conne->getFields("select artquote from tb_article where | 
jd =".$quoteid,0); | 
36 if(!empty($oldid) and !is_null($oldid)){ 
37 S$saveid = $oldid.''.$quoteid : 
38 jelse{ 
39 S$saveid = $quoteid; 
40 b 
41 $sql = "insert into tb_article (typename,author.artquote) values(' 日 记 ',".$name. 
‘,".$saveid.")"; | 
42 Snum = $conne->uidRst($sq]l); | 
43 if($num 一 1){ | 
44 Sreback = "1'; | 
45 }else{ | 
46 Sreback = '3'; | 
47 } 
48 } | 
49 } | 
50 yelse{ | 
51 Sreback = '2'; | 
52 } | 
53 }else if($act == 'delone'){ | 
54 $delonesql = "delete from tb_review where id= ".$_GET['rid']; | 
35 Snum = $conne->uidRst($delonesqD: | 
56 Sreback ="1'; | 
57 jelse if($act =— 'delall){ | 
58 Sdelallsql = "delete from tb_review where artid = ".$_GET['aid']:; | 
59 Snum = $conne->uidRst($delallsq]); | 
60 Sreback = "1'; | 
61 } | 
62 echo S$reback: | 
30 | 


a 


Note 
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11.5 文章 管理 


1 文章 管理 概述 


| 对 于 一 个 博客 系统 来 说 ， 文 章 管理 是 
关中 提供 了 添加 文章 、 文 章 区 表 、 文 章 类 别 等 功能 ， 下 列 分 别 介绍， 
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图 11.17 添加 文章 页 面 的 运行 效果 


11.5.2 ”添加 文章 功能 实现 过 程 


在 添加 文章 页 addart.php 中 创建 一 


本 ， 也 是 最 复杂 的 一 个 功能 。 在 文章 管理 模 


个 表单 ， 其 中 包括 文章 标题 、 文 章 类 别 、 文 字 编辑 


| | 区 、 文 章 内 容 等 表单 元 素 。 该 页 面 的 运行 效果 如 图 11.17 所 示 。 页 面 中 的 部 分 表单 元 素 如 


表 11.1 所 示 。 


表 11.1 添加 文章 页 面 的 主要 元 素 属 性 及 说 明 


元 素 名 称 元 素 属 性 说 明 
text ype="text" id="txt title" size="68" 文章 标题 
<?php 
foreach($typearr as $value){ 
?> 
select <option value="<?php echo $value; ?>"><?php echo $value; ?> </option> | 文章 类 别 
<?php 
?> 
name="font" class="wenbenkuang" id="font” onChange="showfont 
this.options[this.selectedIndex].value)" 奖章 字体 
select class—"wenbenkuang" onChange="showsize(this.options[this.selectedIndex]value)”| 字体 大 小 
al onChange=' aheweolprtiiie ontons[thicselectedinMiee valae) 字体 颜色 
name="color" size="1" class="wenbenkuang" id="select" 
textarea Cols="75" rows="20" 1d="file" style="border:0px:;width:520px;" 文章 内 容 
button id="enterbtn" onClick="return chkartO" “提交 ”按钮 


yf pA 


第 ]| 章 博客 模块 一 一 SS | 
当 用 户 填 写 完 文章 主题 和 文章 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 将 触发 onclick 事件 ， 调 | 
用 chkart0 函数 。 该 函数 将 文章 标题 、 文 章 类 别 等 相关 消息 进行 验证 后 ， 传 递 给 | 
addart_chk php 页 ， 根 据 addart_chk php 页 的 返回 值 进 行 不 同 的 处 理 。chkart0 函 数 存储 于 


jsarticlejs 脚本 文件 中 ， 其 关键 代码 如 下 。 冯 会 内 
01 function chkartO{ | 
02 if(S('txt_title).value=—""){ // 检 测 博客 标题 Note 
03 alert(" 博 客 标题 名 称 不 允许 为 空 ! "); | 
04 SCtxt titlen focusO; | 
05 Teturn false; | 
06 } | 
07 if($('file).value—"){ // 检 测 文章 内 容 | 
08 alert(" 文 章 内 容 不 允许 为 空 ! "); | 
09 SCfile) focus(); | 
10 return false; | 
11 } | 
12 url = 'article/addart_chk.php?act=add&title="+$('txt_title'). | 
13 valuet"&arttype="+$("articletype'). valuet+"&cont=" +$('file').value; /生成 URL | 
14 xmlhttp.open('get',url,true); | 
15 xmlhttp.onreadystatechange = function0){ | 
16 这 xmlhttp readystate 一 4){ | 
17 if(xmlhttp.status 一 200){ | 
18 msg = xmlhttp.response Text:; // 获 取 返 回 值 | 
19 ifmsg = 1){ /如 果 等 于 1， 发 表 成 功 | 
20 alert( 文 章 发 表 成 功 ); | 
21 xmlhttp.open('get','article/article.php',true);// 刷 新 文章 列表 | 
22 xmlhttp.onreadystatechange = functionO{ | 
23 ixmlhttp readystate 一 4 && xmlhttp status == 200){ | 
24 $(0showmenu').innerHITML = xmlhttp.response Text; | 
25 » | 
26 xmlhttp.send(nul]): | 
2 jelse ifmsg 一 '2){ | 
28 alert( 添 加 失败 7; | 
29 jelse{ | 
30 alert(msg); | 
31 3 | 
32 xmlhttp.send(null); ! 
> | 


addart_chk.php 处 理 页 接 到 文章 信息 后 , 将 新 信息 添加 到 数据 库 中 , 如 果 成 功 , 返回 1。 
该 页 的 代码 如 下 : 


01 <?php | 
02 session_ start(); // 开 启 SESSION | 
03 header('content-type:text/html:;charset=gb2312"); // 设 置 编码 | 
04 include "../Conn/conn.php"; // 载 入 数据 库 类 | 
05 Stile=$ GET['title]: // 文 章 标题 | 
06 Stypename = $_GET['arttype']; // 文 章 类 别 | 
07 $author=$ SESSION[mame']: // 当 前 用 户 
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08 $content=$_GET['cont]: 
09 Sreback = "0'; 


10 $sql="Insert Into tb_article (typename ,title,content,author) 
11 Values (".$typename.","". $title.","".$content.",". $author.™)":; //insert 语句 


// 文 章 内 容 


12 Snum = $conne->uidRst($sq]l): // 添 加 文章 

13 ”这 $num 一 ]){ // 如 果 添 加 成 功 

14 $selectsql = "select upfile from tb_member where name = ".$author."" 

15 S$artnum = $conne->getFields($selectsql,0); // 得 到 上 传 文章 数量 
16 $upsql = "update tb_member set upfile = ".(++$artnum)." where name = ".$author."™"; 
I Snum = $conne->uidRst($upsq]); // 更 新 

18 Sreback = "1'; 

19 jelse{ 

20 S$reback = "2'; 

70 

22 echo $reback: 

3 ?> 


”11.5.3 ”文章 列表 功能 实现 过 程 


文章 列表 页 列 出 了 文章 的 类 别 、 标 题 和 点 击 率 ， 并 提供 


| 分 页 等 功能 ， 如 图 11.18 所 示 。 


了 全 选 、 反 选 、 删 除 和 无 刷新 


er & 

| 轩 区 h 入 几 各 世 软 ， 讽 亿 天 大 亿 乐 

| ms 

| a 

| 的 个 人 管理 a [a 大 击 树 

| 总 文章 管理 [ 二 鱼 疝 公告 | 好 联 号 等 水 全 员 t 1 

| Te waie WEe 

| a FR : 

| 品 相遇 党 理 估计 则 | 加 守则 

| 已 好 灵 管 理 二 半 所“ 项 障 法 性 首页 上 一 页 下 一 而 尽责 当 章 县 条 ;页 / 闪 1 责 4 多 记录 出 轩 到 + 1 “| 可 

| 忌 留言 管理 

| 上 十 要 条 

| 图 11.18 文章 列表 页 面 的 运行 效果 

| 文章 列表 页 article php 的 代码 如 下 : 

| 01 ”<?zphp 

| 02 session start(); /开启 SESSION 
| 03 header('content-type:text/html:charset=gb2312"); /设置 编码 

| 04 include_once '../conn/conn.php'; 

| 05 ifisset($_GET[act]){ 1/ 判断 act 参数 

| 06 Sact = $_GET['act']: 

| 07 Jelse{ 

| 08 Sact=""; 

| 09 } 

| 10 $name = $_SESSION['name']:; /获取 当前 用 户 

| 1] Snum = 10; // 每 页 显示 记录 数 
| 12 ifisset($_GET['curpage])){ /判断 是 否 为 当前 页 
| 13 Scurpage = $_GET['curpage']; 

] .274 。 


人 


14 jelse{ | 
15 $curpage =""; | 
1 | 
17 if($act — 'del){ // 如 果 是 删除 操作 | 
18 Srd=$_GET['rd']: // 得 到 要 删除 的 ID 序列 | 
19 $delsql = "delete from tb article where id= -1 ": ”// 初 始 化 删除 语句 | 痊 一 
20 if(!empty($rd)){ /如 果 删 除 序列 不 为 空 | 
2 Sdelarr = explode( ,Srd); // 拆 分 序列 转 为 数组 
2 foreach($delarr as $value){ /循环 数组 | 
23 $delsql = "orid=".$value." "; // 更 新 删除 语句 | 
24 } | 
多 S$delnum = $conne->uidRst($delsq]); // 执 行 删 除 语句 | 
26 } | 
27 } | 
28 if(empty($curpage) or is_null(Scurpage)){ // 如 果 是 第 一 次 打开 该 页 | 
29 Scurpage = 1; // 将 当前 页 变量 设 为 1 | 
30 } | 
31 ifempty($sql) or is_null($sqD){ // 如 果 SQL 查询 语句 为 空 | 
32 // 创 建新 的 查询 语句 | 
33 $sql = "select id,typename,title,hitnum,renum from tb _article where author = ".$name." | 
order by id desc"; | 
34 } | 
35 S$totnum = $conne->getRowsNum($sq]l); // 总 记录 数 | 
36 Stotpage = ceil($totnum / $num); // 总 页 数 | 
37 S$tmpsql = $sql." limit ".($num *($curpage-1)).",".$Snum; // 分 页 | 
38 S$arr = Sconne->getRowsArray($tmpsq]); // 得 到 记录 数组 
39 ?> 


上 述 代码 实现 了 删除 数据 及 分 页 使 用 的 功能 , 接 下 来 就 是 文章 列表 的 显示 。 代码 如 下 : 


01 <div id ='showarticle'> 
02 <table width="600" border="1" cellpadding="0" cellspacing="0" align="center"> 


03 <tr> 

04 <td height="25" width="30" align="center" valign="middle">&nbsp:</td> | 
05 <td height="25" width="100" align="center" valign="middle"> 类 别 </td> | 
06 <td height="25" width="370" align="center" valign="middle"> 标 题 </td> | 
07 <td height="25" width="100" align="center" valign="middle"> 留 言 /点 击 率 </td> | 
08 <ltr> 

09 ”<2php 

10 foreach($arr as $key => $value){ 

11 ?> 

12 <tr> 

ki <td align="center" valign="middle" height="25"> 


14 <input id="chk[<?php echo $key; ?>]" name="chk[]" type="checkbox" value="<?php echo $value 
[ida]; ?>" </td> 


15 <td align="center"” valign="middle"” height=25>&nbsp:;<?php echo Svalu 
e['typename']; ?></td> 
16 <td align="left" valign="middle" height=25>&nbsp;<?php echo $value [ttle]; ?> </td> 


17 <td align="center" valign="middle" height=25>&nbsp:<?php echo $value 
[renum']./.$value[hitnpum']: ?> </td></tr> 


a i 
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删除 、 翻 页 和 跳 转 操作 分 别 调用 了 脚本 文件 choosejs 中 的 3 个 函数 。 删 除 函 数 del0 


| % 
18 <Iphp 
| io 是 
| 以 下 是 全 选 、 反 选 、 删 除 及 分 页 的 实现 ， 其 中 删除 是 全 选 、 反 选 操作 的 延续 。 代 码 如 下 : 
01 if($totnum> 0){ // 如 果 有 记录 
02 ?> 
03 <tr> 
| 04 <td colspan="4" height="30" align="right" valign="middle"> 
| 05 <a href="#" onclick="return alldel(<?php echo count($ar); ?>)"> 全 选 </a> 
| 06 <a href="#" onclick="retum overdel(<?php echo count($ar) ?>")"> 反 选 
| <a> 
| 07 ”<!-- 删 除 操作 调用 del0 函 数 ， 并 传递 了 两 个 参数 --> 
| 08 <button id="delbtn" name="delbtn" class="btn" onclick="retum del(<?php echo 
| $_SERVER['SCRIPT_NAME']: ?>" 
| 09 <?php echo count($arr); ?>',<?php echo $curpage; ?>)"> 删 除 选择 </button> 
| 10 ”<!-- 翻 页 超 链 接 ， 将 调用 pagination0 函 数 ”--> 
| 11 <?php echo ($curpage == 1)?" 首 页 ":" 
| 12 <a onclick='return pagination(\"".$_ SERVER['SCRIPT_NAME']." \",1)> 首 页 </a>"; ?> 
| 13 <?php echo ($curpage 一 1)?" 上 一 页 ":" 
| 14 <a onclick='return pagination(\"".$_SERVER ['SCRIPT NAME']. "\",".($curpage-1).")'> 上 一 
| 页 </a>"; ?> 
| 15 <?php echo ($curpage 一 $totpage)?' 下 一 页 ':" 
| 16 <a onclick=retum pagination(\"".$ SERVER ['SCRIPT_ NAME']."\",".($curpaget+1).")'> 下 一 
| 页 </a>"; ?> 
| 17 <?php echo ($curpage 一 $totpage)?" 尾 页 ":" 
| 18 <a onclick='retum pagination(\"".$ SERVER['SCRIPT NAME']."\",".($totpage).")> 尾 页 
| </a>"; ?> 
| 19 当前 是 第 <?php echo $curpage; ?> 页 / 共 <?php echo S$totpage; ?> 页 <?php echo 
| S$totnum; ?> 条 记录 
| 20 ”<!--” 跳 转 操 作 ， 调 用 jumpmem0 函 数 ”--> 
| 21 跳 转 到 : <select id="jump"” name="jump" onchange="retum jumpmem(<?php echo 
8$_SERVER[SCRIPT_NAMET ?>)'"> 
2 < 
| 23 for($i=1:;$i<=$totpage;$i++){ 
| 24 if($i 一 $curpage) 
| 2 echo "<option value=".$i." selected>".$1."</option>"; 
| 26 else 
| 2 echo "<option value=".$1.">".$1."</option>"; 
| 28 } 
| 29 > 
| 30 </select> 页 </td> 
| 31 </tr> 
| 32 <?php 
| 33 } 
| 34 ?> 
| 35 </table> 
| 36 </div> 
j 
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包含 3 个 参数 , 这 3 个 参数 分 别 代表 删除 处 理 页 的 URL、 当 前 页 的 记录 总 数 和 当前 页 的 页 码 。 

函数 中 首先 声明 了 一 个 数组 rd， 然 后 循环 判断 当前 页 中 的 复 选 框 状 态 ， 将 要 删除 的 复 
选 框 的 vlaue 值 保存 到 rd 中 ; 接着 将 rd 和 当前 页 的 页 码 传 给 处 理 页 ， 最 后 调用 rerstO 函 数 | 
将 处 理 页 的 输出 结果 显示 到 指定 区 域 。choosejs 文件 的 删除 代码 如 下 : 


01 /删除 所 选 

02 function del(chkurl,tot,curpage){ 

03 var rd=new Array(); 1/ 声明 数组 

04 for(i=0j=0;i<tot:it+){ // 循 环 判 断 复 选 框 

05 if(!$('chk["+i+"]'").checked){ | 
06 yelse{ // 如 果 复 选 框 被 选中 | 
07 rd[j] = SCchk[+it])value; /将 value 值 保存 到 rd 数组 中 | 
08 js | 
09 } | 
10 } 

11 iflrd =—"){ // 如 果 没 有 删除 ， 提 示 错 误 

12 alert(' 请 选取 要 删除 数据 !"); 

13 return false: 

14 } 

15 url = chkurlt+"?act=del&rd="+rd+'&curpage='tcurpage; // 生 成 URL 

16 xmlhttp.open('get',url,true); // 创 建新 请 求 

17 xmlhttp.onreadystatechange = rerst; /调用 rerst0 函 数 

18 xmlhttp.send(null); 

19 return false; 

2000 

21 function rerst(){ 

2 if(xmlhttp.readystate 一 4){ 

23 if(xmlhttp.status 一 200){ // 判 断 请 求 是 否 完成 

24 $(showarticle').innerHTML = xmlhttpresponseText /将 返回 值 显示 到 指定 区 域 

25 } 

26 

2 


对 翻 页 的 处 理 统一 使 用 pagination0 函 数 。 该 函数 有 两 个 参数 : 一 个 是 处 理 页 的 URL， 
另 一 个 是 当前 页 的 页 码 。 代 码 如 下 : | 


01 /分 页 

02 function pagination(chkurl,curpage){ ! 
03 url = chkurl+'?act=next&curpage='+curpage; // 生 成 URL | 
04 xmlhttp.open('get',url,true): // 创 建新 请 求 

05 xmlhttp.onreadystatechange = rerst: // 调 用 rerst0 函 数 

06 xmlhttp.send(null); | 
07 return false: | 
08 } | 


跳 转 是 翻 页 的 另 一 种 形式 。 翻 页 只 能 一 页 一 页 地 翻 , 而 跳 转 可 以 直接 跳 到 指定 的 页 面 。 
代码 如 下 : 
01 “”/W/ 跳 转 


A 


AI 


function jumpmem(chkurD){ 
jumppage = $(jump").value; // 获 取 跳 转 页 数 
url = chkurl+ ?curpage=+jumppage: // 生 成 URL 
xmlhttp.open('get',url,true); // 创 建 请 求 
xmlhttp.onreadystatechange = rerst; /调用 rsrst0 函 数 
xmlhttp.send(nul]); 
return false; 

} 


”11.5.4 ”文章 类 别 功能 实现 过 程 


在 文章 类 别 模块 中 可 以 实现 文章 类 别 的 添加 、 删 除 。 文 章 类 别 页 面 的 运行 效果 如 


图 11.19 所 示 。 
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图 11.19 文章 类 别 页 面 的 运行 效果 
文章 的 类 别 数据 被 保存 到 用 户 信息 表 的 一 个 类 别 字段 中 。 默 认 有 一 个 “日 记 ” 类 别 ， 


当 有 多 个 类 别 时 ， 将 使 用 逗号 分 割 。 显 示 文 章 类 别 时 ， 首 先 获取 到 该 用 户 信 息 中 类 别 字段 


| 的 值 ， 将 字符 串 使 用 逗号 分 割 成 数组 ， 最 后 由 foreach 循环 显示 所 有 类 别 。 文 章 类 别处 理 


| 文件 arttype.php 的 代码 如 下 : 
| oO Zplip 
02 session start(); /开启 SESSION 
03 header('content-type:text/html:charset=gb2312"); /设置 编码 
04 include_once ' /conn/conn .php': // 载 入 数据 库 类 文件 
05 $act=$_GET['act]: /获取 act 值 
06 $name = $_SESSION['name']; 1/ 获取 用 户 名 
07 $sql = "select arttype from tb_member where name = ".$name.""; ”// 生 成 查询 语句 
08 Stypefields = $conne->getFields($sql,0); // 获 取 字 段 值 
(00 > 
10 <table id="addtypefim" width="300" border="1" cellspacing="0" cellpadding="0" align= 
| "center"> 
je <tr> 
和 <td colspan="2" height="25" align="center" valign="middle"> 文 章 类 别管 理 </td> 
13 </tr> 
14 <?php 
15 Stypearr = explode(',', $typefields): /使 用 喜 号 分 隔 字符 串 存 为 数组 
16 foreach($typearr as $value){ /循环 输出 数组 
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21 <button id="deltype" value="<?php echo $value; ?>" class="btn" 
22 onclick="return delarttype('<?php echo S$typefields; ?>''<?php echo $value;?>")"> 删 除 
</button></td> 


17 > 
18 <tr> 
19 <td width="200" height="25" align="center"” valign="middle">&nbsp;<?php echo | 
S$value; ?></td> | 
20 <td height="25" align="center" valign="middle">&nbsp:; | 优 站 


23 </tr> | 
24 <?php | 
25 } | 
26 ?> | 
27 <tr> | 
28 <td width="200" height="25" align="center" valign="middle"> | 


29 <input id="addarttype" name="addarttype" type="text" class="txt" /></td> | 

30 <td height="25" align="center" valign="middle"><button id="addtype" value="<?php echo | 
$value; ?>" class="btn" | 

El | onclick="return addarttype('<?php echo $typefields; ?>")"> 添 加 新 类 </button></td> 

32 </tr> 

33 </table> 


如 要 添加 文章 类 别 ， 单 击 “ 添 加 新 类 ”按钮 后 ， 将 调用 addarttype() 函 数 ， 并 将 原 类 别 
字段 传 给 该 函数 。addarttype0 函 数 将 创建 一 个 新 请 求 ， 并 根据 请 求 的 输出 值 进行 不 同 的 处 | 
理 。addarttype() 函 数 存 储 于 js\article.js 脚本 文件 中 ， 其 代码 如 下 : | 


01 /添加 文章 类 别 
02 function addarttype(typestD{ 


03 ifSCaddarttype) value — "){ /判断 类 别名 称 是 否 为 空 
04 alert(' 请 输入 添加 类 别名 称 '); 
05 S$('addarttype').focusO; 
06 Teturn false; 
07 } | 
08 ul = | 
"article/arttype_chk.php?act=add&typestr="+typestr+'&typename="+$('addarttype').value ; | 
09 xmlhttp.open('get',url,true); // 创 建新 请 求 | 
10 xmlhttp.onreadystatechange = function0{ | 
11 if(xmlhttp.readystate 一 4){ | 
12 if(xmlhttp.status 一 200){ | 
13 msg=xmlhttp responseText: // 获 取 处 理 页 输出 结果 | 
14 // 根 据 返 回 的 不 同 结果 进行 不 同 的 处 理 | 
15 if(msg — 2){ | 
16 alert( 添 加 失败 ); | 
17 }else itmsg — 3){ | 
18 alert( 添 加 名 称 重复 ): | 
19 jelseifmsg 一 '1){ | 
20 showarttype(); | 
2 jelsef | 
2 alert(msg); | 
23 n | 
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其 代码 如 下 : 


.Opp 内 齐 搞 半 精 角 


xmlhttp.send(nul]); 
} 


arttype_chk.php 是 addarttype() 函 数 调用 的 处 理 页 ， 将 获取 到 的 原 类 别 字段 后 面 加 上 去 
号 和 新 类 别名 称 ， 再 将 该 字段 保存 。 其 代码 如 下 : 


过 


<?php 
session start(); 
header('Content-Type:text/html:charset=gb2312"): 
include_once '../../conn/conn.php'; 
$act=$_GET['act]; 
$typestr =$_GET[typestr]: 
$typename =$_GET['typename']; 
S$arr = explode(',', $typestr); 
S$reback = '0'; 
$name = $_SESSION['name']; 
if($act == 'add){ 
if(in_array($typename,$arr)){ 
Sreback = '3"'; 
jelse{ 
S$typestr .=",".$typename: 


/开启 SESSION 支持 
/设置 编码 

// 载 入 数据 库 类 

// 获 取 act 值 

// 原 类 别 字符 串 

// 要 获取 的 类 别 字 符 串 
/将 原 字符 串 转换 为 数组 


/获得 当前 用 户 


/如 果 是 添加 操作 
// 判 断 名 称 是 否 重复 


// 将 新 类 别名 称 添加 到 原 类 别 后 面 


$sql = "update tb member set arttype = ".$typestr." where name = 


$num = $conne->uidRst($sq]); 
if($num 一 1){ 
Sreback = "1'; 
}else{ 
Sreback = 2'; 
}s 


// 执 行 更 新 语句 


| 删除 文章 类 别 和 添加 文章 类 别 大 同 小 异 ， 但 要 注意 两 点 。(1) 不 允许 删除 默认 的 “日 
| 记 ” 类 别 ， 如 果 删 除 的 是 “日 记 ” 类 别 ， 则 弹出 警告 ，(2) 将 删除 的 类 别 下 的 所 有 文章 ， 
| 全 部 放 到 默认 的 “上 日记” 目录 下 。 删 除 文章 类 别 操作 是 在 arttype_chk php 文件 中 完成 的 ， 


01 
02 


// 接 添加 类 别 代码 
else if($act — 'del){ 
if($typename =— ' 日 记 ){ 
Sreback = 3'; 
jelse{ 
$tmpstr = ' 日 记 ': 
for($i=1:$i<count($arr):$i++){ 
if($arr[$1] != $typename){ 
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09 Stmpstr =",".$ar [$i]; 
10 } 


| 
12 Schangsql= "update tb _article set typename = ' 日 记 'where author =".$name." and typename= | 


13 Snum = $conne->uidRst($changsq]); | 一 

14 S$sql = "update tb member set arttype ="".$tmpstr." where =" Sname.™; 

15 $num = $conne->uidRst($sq]); 
16 if($num 一 D){ | 

17 Sreback ="1'; | 

18 jelse{ 

19 Sreback = '2'; 

20 让 

21 } | 


} 
3 echo S$reback: 


11.6 用 户 管 理 


11.6.1 用户 管 理 概述 


博客 后 台 管理 系统 主要 由 六 大 模块 构成 ， 分 别 是 用 户 管理 模块 、 文 章 管理 模块 、 图 片 管 
理 模块 、 管 理 员 设置 模块 、 系 统 日 志 模块 以 及 数据 备份 模块 。 本 节 重 点 讲解 用 户 管理 模块 ， | 
其 运行 效果 如 图 11.20 所 示 。 


Pa 
9 日 #0 pH 
a 要 4 hasH 
0 于 4 本 Cla) 
3 于 #8 pa 
s 失 芝 让 pas 
s 日 2 iaH 
s 日 4 RH 


首页 上 一 而 下 -页 局 当前 是 第 1 页/ 失 1 页 条 记录 只 到 ; ! ”而 


所 | 拉 六 枯 如 t 扫 ;nt3i-oeT6s6| 传 直 : D43I-84sT2zt6 生 业 HE 葵 3 nireri suntaerisoft onn 
Corp owe ardead cen Ml Lieds ecervedl 


图 11.20 后 台 用 户 管理 模块 运行 效果 
顾名思义 ， 用 户 管理 模块 就 是 用 来 管理 注册 用 户 的 。 在 该 模块 中 提供 了 多 项 功能 ， 如 | 
查询 博客 、 推 荐 博客 、 冻 结 博客 、 查 看 详细 资料 和 删除 博客 等 。 其 中 推荐 博客 和 冻结 博客 
的 实现 方法 基本 相同 ， 而 查看 详细 资料 和 删除 博客 在 前 面 也 接触 过 ， 这 里 只 讲解 查询 博客 | 
和 推荐 博客 的 实现 方法 。 | 
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VAI 


可 
11.6.2 用 户 管理 功能 实现 过 程 


| 首先 来 看 一 下 用 户 管理 模块 的 首页 memberphp。 在 memberphp 文件 中 ， 将 数据 库 中 


| 存储 的 用 户 信息 进行 分 页 输出 ， 并 且 完 成 各 种 附加 功能 的 设置 。 其 运行 效果 如 图 11.21 所 示 。 
站 | 编号 。。 用 PUK 呈 nail 下 村 名 性别 。 坷 多 5 市 计 扒 天 寺 时 / 亢 车 主 深 科 
加 和 5 Te 更 o EE 车 说 江 料 
= a 。 摊 。ih 
| | 2 jzlll EE o 否 东 结 次 村 
| 回 22 a 江 济 档 榜 TT8452eaa eon 王仁 画 让 3 3 车 铺 详 沪 料 
| aa| 博客 皇帝 itlige* < 张大 PE 男 下 淮 泛 夺 结 详细 资料 
回 20 yo Tzse504009e cm 小 页 田 ES 本 4 详 旺 资料 
下 19 Btslzg549qe rm 123 女 5 区 车 这 料 
主流 E 寺 “是 出 际 半 撕 首页 上 一 页 下 一 页 局 当前 是 浓 ! 页 /站 1 页 + 条 记 录 中 四 到 : 1 ”页 
图 11.21 用 户 管理 页 面 的 运行 效果 
memberphp 页 的 代码 如 下 : 
01 <?php 
02 session start(); /开启 SESSION 
03 include_once '../conn/conn.php'; // 载 入 数据 库 类 
04 ?> 
05 <script language="javascript" sre="../js/choose.js"></script> ”// 载 入 JS 文件 
06 <?php 
07 Snum=10; // 每 页 显示 记录 数 
08 iflempty($_GET['act])){ /获取 当前 是 第 几 页 
09 Sat="": 
10 }else{ 
11 S$act=$_GET['act]; // 当 前 页 
i } 
13 $sql=""; //SQL 语句 
14 if(empty($curpage) or is_null($Scurpage)) { 1/ 获取 当前 是 第 几 页 
15 S$curpage = 1; 
16 jelse{ 
I S$curpage = $_GET['curpage']; // 当 前 页 
18 及 
19 if(empty($fields) or is_null(Sfields)){ // 排 列 顺序 ， 默 认为 id 
20 Sfields = "id'; 
21 }else{ 
22 Sfields = $_GET['fields']; /依照 该 字段 排列 记录 
23 } 
24 // 是 否 有 查询 内 容 ， 如 果 没 有 
25 ilisset($_REQUEST['cont]){ 
| 26 // 直 接生 成 SQL 查询 语句 ， 使 用 $fields 字段 作为 排序 依据 
| 27 $sql = "select id.name,email.realname,sex,birthday.hitnum ,freeze,isnominate from tb member order 
by ".Sfields." dese"; 
| 28 // 如 果 有 查询 内 容 ， 就 获取 要 查询 的 字段 、 查 询 方 式 和 查询 内 容 
| 29 Jelse{ 
| 30 Squerymem = $ REQUEST['querymem ']: 
| 31 Ssignslt =$ REQUEST[signslt]: 
32 Scont=$ REQUEST[cont]: 
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11.6.3 ”查询 博客 功能 实现 过 程 


查询 博客 是 根据 博客 不 同 的 信息 来 查询 特定 的 博客 ， 如 ID 号 、 点 击 率 等 。 查 询 页 是 
单独 的 一 页 ， 被 包含 在 memberphp 页 中 。 当 用 户 查询 时 ， 该 页 将 查询 字段 、 查 询 方式 和 
查询 内 容 传 给 memberphp 页 ， 再 由 member.php 页 进行 处 理 (member.php 页 的 处 理 过 程 在 
上 面 已 经 介绍 过 )。 查 询 页 query.php 的 代码 如 下 : 


33 /初始 化 变量 $sql 为 一 个 无 条 件 的 查询 语句 

34 $sql = 'select id,name,email.realname.sex,birthday.,hitnum,freeze,isnominate from 
tb_ member'; 

35 // 根 据 查询 字段 所 在 的 组 ， 生 成 不 同 的 where 子 句 

36 if(in array($querymem.,array('id',"upfile','uppics','hitnum','birthday"))){ 

3 S$sql = "where ".$querymem." ".$signslt." ".$cont; 

38 }else if(in array($querymem,array(‘name','realname','qq','email’))){ 

39 if($signslt 一 'exac){ 

40 $sql = "where ".$querymem." =".$cont.""; 

41 }else if($signslt 一 ‘mist){ 

42 $sql = "where ".$querymem." like '%".$cont."%"; 

43 } 

44 } 

45 // 添 加 order by 子 句 

46 $sql = " order by ".$fields." desc"; | 

47 } | 

48 S$totnum = $conne->getRowsNum($sq]); // 记 录 数 | 

49 S$totpage = ceil($totnum / $num); // 页 数 | 

50 S$tmpsql = $sql." limit ".($num *($curpage-1)).",".$num; | 

| Sarr = $conne->getRowsArray($tmpsq]); | 

52 ?> | 


01 <?php 
02 include_once '../conn/conn.php'; // 载 入 数据 库 类 
03 ?> 


04 <script language="javascript" sre="../js/query.js"></script> 

05 <table border="0" cellpadding="0" cellspacing="0" align="center"> 

06 <fom id="queryfm" name="queryfm" method="post" action="memberphp?act=query" 
onSubmit="returm chkqueryO'> 


07 <tr> | 
08 <td width="50" height="25"> 查 询 : </td> | 
09 <td width="100"> | 
10 <select 1d="querymem" name="querymem" onChange="return changequeryO"> | 
11 <option value="id"> 编 号 </option> | 
jb <option value="name"> 账 号 </option> | 
13 <option value="realname"> 真 实 姓名 </option> | 
14 <option value="email">email</option> | 
15 <option value="qq">qq</option> | 
16 <option value="birthday"> 出 生日 期 </option> | 
17 <option value="upfile"> 文 章 数 </option> | 
18 <option value="uppics"> 上 传 图 片 </option> | 
四 
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人 > 


<option value="hitnum"> 点 击 率 </option> 
</select> 
</td> 
<td width="50" align="center" valign="middle"><div id="sign"> 
<select 1d="signslt" name="signslt"> 
<Option value="&gt:"> 大 于 </Option> 


<option value="="> 等 于 </option> 
<option value="&lt;"> 小 于 </option> 
</select> 
</div></td> 


<td width="100"><div id="input"><input id="cont" name="cont" type="text" 


| size="30" /></div></td> 


| 30 
| 31 
| 32 
| 33 


<td><input id="querybt" name="querybt" type="submit" value=" 查 询 "></td> 


</tr> 
</form> 
</table> 


在 查询 页 中 ， 当 用 户 选 择 不 同 的 查询 条 件 时 ， 查 询 方 式 也 会 随 之 发 生变 化 。 例 如 查询 


| ID、 点 


率 时 ， 查 询 条 件 有 大 于 、 等 于 和 小 于 ， 当 查询 账号 、 真 实 姓 名 、E-mail 时 ， 查 询 


条 件 变 为 精确 和 模糊 。 这 是 通过 queryjs 脚本 来 控制 的 。 实 现代 码 如 下 : 
| 01 function changequeryO{ 


| 02 key = document.getElementById('querymem').value; // 获 取 查 询 条 件 
| 03 switch(key){ 
| 04 人 # 当 查询 条 件 和 数字 有 关系 时 所 
| 05 case 'id': 
| 06 case "upfile': 
| 07 case "uppics': 
| 08 case hitnum': 
| 09 document.getElementById('sign).innerHTML = '<select id="signslt" name= 
| "signslt"> 
| 10 <Option value="&gt:"> 大 于 </Option><option value="="> 等 于 </option><option value="&lt;"> 小 于 
| </option></select>"; 
| 11 document.getElementById('cont").value="; 
| 12 break: 
| 13 庶 当 查询 条 件 和 字符 串 有 关系 时 */ 
| 14 case ‘name': 
| 15 case Tealname'- 
| 16 case 'email : 
| 17 case 'qq': 
| 18 case ‘title': 
| 19 case 'author: 
| 20 case 'typename': 
2 case ‘upname': 
| 2 case "upauthor- 
| 23 case 'operator - 
| 24 document.getElementByld('sign').innerHTML = '<select id="signslt" name= 
| "signslt"> 


2 <Option value= "exac"> 精 确 </Option><option value="mist"> 模 糊 </option></select>'; 
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26 document.getElementById('cont").value=": | 
2 break: | 
28 局 当 查 询 条件 和 时 间 有 关系 时 */ | 
29 case ‘birthday’: | 
30 case 'firsttime': ! 大 
3 case "uptime' | 到 J 
32 document.getElementById('sign').innerHTML = '<select id="signslt" name= | 
"signslt"> Note 


33 <Option value="&gt;"> 大 于 </Option><option value="="> 等 于 </option><option value="&lt:"> 小 于 | 
</option></select>'; | 


34 document.getElementById(cont).value=' 查 询 格式 : YYYY-MM-DD'; | 
35 break: | 
36 /# 省 略 部 分 为 其 他 模块 的 查询 方式 设置 #/ | 
37 EE | 
38 } | 
39 document.getElementById('cont').focus(); 

40 document.getElementById(cont).selectO: 

A700 


11.6.4 ”推荐 博客 功能 实现 过 程 


当 用 户 为 推荐 博客 时 ， 该 用 户 的 博客 空间 将 在 首页 的 显著 位 置 显示 ， 这 样 可 以 提高 网 
站 和 个 人 空间 的 人 气 和 访问 量 。 推 荐 博客 实现 方法 很 简单 ， 在 用 户 表 中 设置 一 个 字段 ， 名 
称 是 isnominate， 类 型 为 int。 当 用 户 注册 时 ， 字 段 的 默认 值 为 0; 当 设 为 推荐 博客 时 ， 将 
该 字段 设 为 1。 该 操作 从 memberphp 文件 中 设置 的 超 链 接 开 始 ， 其 代码 如 下 。 


<a href="member chk.php?act=nominate&id=<?php echo $value['id]: ?>&isnominate=<?php echo 
S$value [isnominate']; ?>"><?php echo ($value['isnominate']-=0?" 否 ":" 推 荐 "); ?></a> 


链接 到 member_chk.php 文件 ， 根 据 博 客 用 户 的 ID 号 和 isnominate 字段 的 值 来 更 新 用 
户 表 。 其 代码 如 下 : 


01 <?php | 
02 session start(); | 
03 header('Content-Type:text/html;charset=gb2312"): | 
04 include_once '../conn/conn.php'; | 
05 $act = $_GET['act]: | 
06 if($act == 'del){ | 
和 和 1/ 删除 操 作 | 
08 jelse if($act 一 nominate ){ /推荐 博客 | 
09 $id=$_ GET[id]: /获取 用 户 ID 号 | 
10 Snominate = ($_GET[isnominate]+1) % 2: // 将 获取 的 字段 值 加 1 再 | 

模 2 | 
11 $sql = "update tb_member set isnominate = ".$nominate." where id = ".$id; | 
12 Scont = "推荐 用 户 操作 ， 账 号 : '$id: /系统 日 志 内 容 | 
13 Snum = $conne->uidRst($sq]): // 更 新 字段 值 | 
14 if(!empty(Snum)){ 


“289%s 


15 Sinssql 三 "insert 
values(",".$cont.",".$_SESSION['manager].",nowO)":; 


16 Snum = $conne->uidRst($inssq]l); 
17 } 
18 echo "<script>location="member.php';</script>"; 
19 } 
11.7 技术 提炼 


11.7.1 文件 包含 技术 


| 通过 文件 包含 技术 实现 文件 的 包含 ， 有 助 于 网 站 页 面 的 规划 、 管 理 和 更 新 。 例 如 ， 网 
”站 的 头 文件 、 尾 文件 和 数据 库 的 连接 文件 ， 都 可 以 通过 包含 的 方式 进行 载 入 ， 这 样 就 避免 


了 在 每 个 页 面 中 编写 相同 的 代码 ， 减 少 了 代码 的 元 余 ， 同 时 也 便于 对 这 些 内 容 的 修改 和 更 新 。 
| 在 博客 模块 的 个 人 首页 中 通过 include_once() 函 数 包含 文件 。include_once() 函 数 应 月 
| 脚本 执行 期 间 同一 个 文件 有 可 能 被 包含 超过 一 次 的 情况 下 ， 以 确保 它 只 被 包含 一 次 ， 从 而 


tb log 


:3 


| 避免 出 现 函 数 重 定义 、 变 量 重新 赋值 等 问题 。 也 就 是 说 ,使 用 include_once() 函 数 包含 文件 
| 时 ， 如 果 该 文件 中 的 代码 已 经 被 包含 ， 则 不 会 被 再 次 包含 。 


下 面 是 博客 个 人 首页 的 左 侧 包含 文件 代码 ; 


01 <!-- 左 侧 --> 

02 <div id="leftfloatcont"> 

03 <div id="personinfo"> 

04 <?php include_once 'personinfo.php'; ?> 
05 </div> 

06 <div id="date"> 

07 <?php include_once 'date.html'; ?> 
08 </div> 

09 <div id="showmessdiv"> 

10 <?php include_once 'messl.php'; ?> 
11 </div> 

12 </div> 


“11.7.2 ”SQL 查询 技术 


在 “我 的 文章 ”模块 中 ， 总 体 是 采用 SQL 语句 查询 技术 ， 从 数据 库 中 根据 查询 条 件 


完成 数据 的 循环 输出 。 其 中 用 到 的 SQL 技术 如 下 : 
1， 排 序 功 能 


排序 分 为 升序 和 降序 ， 升 序 是 从 小 到 大 ， 而 降序 则 相反 ， 是 从 大 到 小 。 排 序 主要 有 两 


个 关键 字 : 升序 asc、 降 序 desc。 其 语法 格式 如 下 : 
select * ffom 表 名 order by 字段 名 asc/desc 
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2. 查询 记录 总 数 


将 数据 库 中 的 记录 总 数 查 询 出 来 ， 
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使 用 的 关键 字 是 count 和 as。 其 语法 格式 如 下 : 


select count (*) as ( 自 定 义 字 段 ) from 表 名 


3. 条 件 查询 


按照 指定 的 一 个 或 多 个 条 件 进行 查询 ， 使 用 的 关键 字 是 where。 其 语法 格式 如 下 : 
select * fom 表 名 where 字段 = 条 件 
下 面 讲解 一 下 这 些 SQL 查询 技术 在 本 模块 中 的 具体 应 用 。 


(1)“ 我 的 文章 ”中 应 用 的 查询 语句 ， 其 代码 如 下 : 


01 $sql = "select * from tb_article where author =".$_ GET[uid']." order by id desc "; 
02 S$totnum = $conne->getRowsNum($sq]); 

03 S$totpage = ceil($totnum / $num); 

04 S$tmpsql= $sql." limit ".(Snum *($curpage-1)).",".$num:; 

05 $arr = $conne->getRowsArray($tmpsq)l); 


06 S$conne->close_rst(); 


查询 文章 表 (tb_article》 中 的 所 有 记录 ， 查 询 条 件 是 author 的 值 与 超 链接 传递 的 值 相 


同 ， 并 以 ID 为 条 件 进行 降 浓 排列 。 最 后 输出 查询 结果 ， 完 成 我 的 文章 标题 、 发 表 时 间 、 


部 分 文章 内 容 的 循环 输出 。 


(2) 文章 浏览 中 应 用 的 查询 语句 ， 其 代码 如 下 : 


01 $showartsql = "select * from tb_article where id = ".$_GET['artid']; /生成 查询 语句 
02 Sartarr= $conne->getRowsArray($showartsql):; // 返 回 数组 形式 的 记录 


03 Sconne->close_rst(); 


以 超 链 接 中 传递 的 文章 的 ID 为 条 件 ， 执 行 查 询 操作 ， 查 询 的 结果 就 是 有 关 这 条 记录 


的 详细 信息 ， 即 文章 的 详细 内 容 。 
11.7.3 ”会 话 处 理 技术 
想 要 使 用 文章 管理 模块 ， 前 提 是 


要 删除 文章 和 评论 ， 前 提 是 当前 用 户 要 么 是 管理 员 ， 要 么 是 文章 拥有 者 ， 否 则 不 会 显示 删 | 
除 功能 。 这 两 方面 的 控制 都 需要 SESSION 的 配合 ， 本 节 就 来 讲 一 下 SESSION 的 应 用 及 常 | 


见 的 问题 处 理 。 


SESSION 的 中 文 含义 为 “会 话 ”， 是 指 用 户 从 进入 网 站 开始 ， 直 到 关闭 网 站 这 上段 时 间 
内 ， 所 有 网 页 共同 使 用 的 公共 变量 的 存储 机 制 。SESSION 比 Cookie 更 有 优势 : SESSION | 
是 存储 在 服务 器 端的 ， 不 易 被 伪造 ，SESSION 的 存储 没有 长 度 限 制 ，SESSION 的 控制 更 | 


处 理 函 数 如 表 11.2 所 示 。 


/清空 结果 集 


有 户 必须 登录 ， 匿 名 用 户 是 无 法 访问 这 些 功 能 的 ， 想 | 


PHP 主要 是 通过 会 话 (SESSION) 处 理 函 数 来 对 SESSION 进行 控制 和 使 用 。 常 用 的 
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表 11.2 PHP 常用 的 会 话 处 理 函 数 


功能 说 明 
开启 SESSION 或 返回 已 经 存在 的 SESSION 
注册 一 个 SESSION 变量 
设 定 或 取得 当前 的 session_id 值 
检测 指定 的 SESSION 值 是 否 存在 。Isset 不 仅 可 以 检测 SESSION， 还 可 以 
检测 其 他 类 型 ， 如 isset($_post['name'])、isset($_get['mname"]) 等 
更 改 session id 的 值 

返回 或 改变 当前 SESSION 的 name。 如 果 要 改变 当前 SESSION 的 name 值 ， 
必须 在 session0 之 前 调用 session_name0 函 数 , 而 且 session name 不 能 全 部 
是 数字 ， 否 则 会 不 停 地 生成 新 的 session_id 

删除 名 为 name 的 SESSION。 注 意 ， 不 能 写成 unset($_SESSION)， 否 则 会 
禁止 整个 会 话 的 功能 
结束 当前 会 话 ， 删 除 所 有 session 


下 面 介绍 几 个 使 用 SESSION 时 要 注意 的 问题 。 
1. 尽 可 能 地 将 session_start 放 到 第 1 行 
这 种 情况 是 新 手 最 容易 犯 的 错误 。 产 生 的 错误 代码 为 : 
Warning: session start() [function.session-start]: Cannot send session cache limiter - headers already sent… 


其 原因 就 是 在 使 用 session_start0 之 前 ， 有 HTML 代码 输出 。 也 许 有 的 读者 会 说 : 
session_start 之 前 没有 任何 代码 ， 绝 对 没有 ,但 还 是 出 现 了 这 个 错误 。 此 时 应 检查 程序 ， 看 
看 是 不 是 有 空 行 或 类 似 echo 语句 的 输出 。 如 果 有 ， 则 去 掉 ， 因 为 就 算是 一 个 小 小 的 空格 
都 是 不 可 以 的 。 为 了 避免 这 类 错误 的 发 生 ， 应 尽 可 能 地 将 session_start 放 到 第 1 行 。 


2. 在 使 用 SESSION 之 前 一 定 要 先 写 session start() 


大 多 数 读者 在 使 用 SESSION 之 前 都 能 先 调 用 session start0) 函数 ， 但 对 于 
session_destroy() 函 数 却 经 常 忽略 。session destroy0) 虽 然 是 结束 当前 会 话 并 删除 所 有 


session start(); 
$ SESSION[mame' 


lsset($_ SESSION[mname]) 


session regenerate id() 


session name() 


unset($_SESSION['name']) 


session_destro 


”SESSION， 但 在 删除 之 前 ， 也 要 先 开启 SESSION 支持 才 可 以 ， 不 然 会 产生 如 下 的 错误 代码 。 


session_destroy() [function.session-destroy]: Trying to destroy uninitialized session in… 

所 以 ， 凡 是 在 使 用 SESSION 函数 的 页 面 中 ， 都 要 加 上 session start()。 

3. 删除 所 有 SESSION 

如 果 想 删除 所 有 SESSION， 但 又 不 想 结束 当前 会 话 ， 用 unset0 一 个 一 个 删除 实在 是 太 麻烦 
了 ， 最 简单 的 办 法 就 是 将 一 个 空 数组 赋 给 $ SESSION， 如 $_SESSION = array0， 这 样 就 解决 了 。 
11.7.4 ”冻结 、 解 冻 技术 


所 谓 用 户 冻 结 技术 ,就 是 将 用 户 冻 结 后 ,他 将 无 法 登录 。 有 冻 自 然 就 有 解 ， 如 果 解 冻 ， 
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用 户 便 可 以 正常 登录 。 用 户 冻 结 技术 主要 是 针对 一 些 不 守 网 站 规则 的 用 户 而 设立 。 其 实用 | 
户 冻 结 技术 与 用 户 推荐 、 文 章 审 核 等 技术 的 实现 原理 是 相同 的 。 | 
采用 数据 库 条 件 式 方法 ， 在 指定 数据 库 内 设置 单独 的 字段 作为 冻结 技术 的 条 件 ， 当 条 
件 为 真 时 用 户 便 是 正常 运行 (解冻 )， 如 果 条 件 为 假 用 户 便 停止 运行 (冻结 )， 这 就 是 冻结 
和 解冻 的 原理 。 本 模块 中 具体 的 冻结 、 解 冻 操作 是 在 adtminmembermemberphp 文件 中 完 | 
成 的 ， 其 代码 如 下 : 


01 sid=$ GET[id]; 


02 Sfreeze = ($_GET['fz"]+1) % 2; 
03 $sql = "update tb_ member set freeze = ".$freeze." where id = ".$id; | 
04 $cont = ' 冻 结 (解冻 ) 操 作 ， 账 号 : '.$id; | 
05 $num = $conne->uidRst($sq]l); | 
06 if(!empty($num)){ 
07 S$inssql = "insert into tb_log values(",".$cont.",".$_SESSION['manager].",nowO)"; 
08 Snum = $conne->uidRst($inssqD: 

11.7.5 配置 文件 


在 配置 文件 中 设置 了 服务 器 的 目录 、 博 客 目 录 、 上 传 图 片 目录 、 图 片 大 小 上 限 等 信息 。 
配置 文件 config php 存储 于 前 台 。 配 置 文件 代码 如 下 : 


01 <?php 

02 define(PATH',$_ SERVER[IDOCUMENT ROOT]):; /服务 器 目录 

03 define(‘ROOT',/db_blog1.0 由:; /博客 目录 

04 define(ADMIN','admin/"):; // 后 台 目 录 

05 define('PIC','center/pics/image/"); /上 传 图 片 目录 

06 define('BAK,','sqlbak/"); 1/ 备份 目录 

07 define(HEADGIF','headgif/"): // 头 像 目 录 

08 define(MYSQLPATH','D:\xampp\\MySQL\\bin\): /MySQL 执行 文件 路 径 

09 define(MYSQLHOST','127.0.0.1"); /MySQL 服务 器 卫 

10 define(MYSQLDATA'db_blog)); /MySQL 数据 库 | 
11 define(MYSQLUSERwroot); /MySQL 账号 | 
2 define(MYSQLPWD',111"); /MySQL 密码 | 
13 define(MAXSIZEPIC',500000): /允许 上 传 的 图 片 的 最 大 字 节 数 | 
14 和 | 


11.7.6 ”公共 函数 


博客 系统 的 公共 函数 有 两 个 用 途 : 一 是 返回 指定 文件 夹 下 的 文件 列表 ; 二 是 判断 上 传 | 
文件 的 后 级 。 公 共 函 数 存储 于 centerinc\func php， 其 代码 如 下 : | 
01 <?php 


02 ”/W/ 返 回 文件 夹 下 的 文件 列表 | 
03 function show_file($f name){ 


04 $d_open = opendir($f name): /打开 指定 目录 | 
05 $num = 0; 
06 while($Sfile = readdir($d_open)){ 1/ 循环 输出 目录 下 文件 
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| 07 Sfilename[$num] = $file; // 将 文件 保存 成 数组 
| 08 Snumtt+; 
| 09 } 
| 10 closedir($d_open); // 关 闭 目录 
付 办 | 11 return $filename; // 返 回 文件 数组 
一 | 12 
| 13 户 
Note 14 “+ 判断 文件 后 级 


| 15 *Sf type: 允许 的 文件 后 级 类 型 
16 *Sf upfiles: 上 传 文件 名 


| 17 所 

| 18 functionf postfix(Sf type,Sf upfiles){ 

| 19 Sis_pass = false; 

| 20 Stmp_upfiles = split("™\.",$f_ upfiles); // 将 后 级 字符 串 分 给 数组 
| 21 Stmp_num = count(Stmp_ upfiles): // 获 取 数组 大 小 

| 22 if(in_array(strtolower($tmp_upfiles[$tmp_num - 1]).$f type)) // 上 传 的 后 缀 是 否 
”在 允许 范围 内 

| 23 Sis_pass = $tmp_upfiles[Stmp_num - 1]; /保存 后 组 

| 24 return $is_pass; // 返 回 结 果 

| 25 汪汪 

| 26 ?7> 
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在 线 编程 词典 模块 
( 铭 d 自学 视频 、 源 程序 : 配套 资源 \mmN12A ) 


“在 线 编 程 词典 ”是 明日 科技 编程 词典 软件 的 网 页 版 ， 实 现 了 编程 词典 软 
件 基 本 的 浏览 功能 ， 其 目的 是 让 浏览 者 通过 网 页 体会 编程 词典 软件 的 海量 数据 
资源 给 编程 带 来 的 极 大 便利 。 

通过 本 章 的 学 习 ， 读 者 将 体会 到 一 种 全 新 的 编程 方式 ， 不 是 单纯 地 为 基 个 
企业 开发 一 个 网 站 ， 而 是 将 企业 开发 的 基 个 软件 以 网 页 的 形式 展示 给 大 家 。 在 
这 种 全 新 的 背景 下 ， 学 习 程 序 开发 的 技术 、 技 巧 ， 积累 开发 经 验 。 


DI 如 何 设计 树 状 导 航 莱 单 

”如何 向 Flash 中 传递 参数 

也 ”如何 用 Smarty 注册 模板 函数 一 一 读 取 文件 
四 了 解 PDO 数据 库 抽 象 层 

MW 超 链 接 中 的 图 片 切 换 技 术 


WA 


12.1 在 线 编程 词典 概述 


12.1.1 ”模块 概述 


编程 词典 是 由 吉林 省 明日 科技 有 限 公 司 研发 的 一 款 编程 辅助 软件 ， 是 明日 科技 数 年 开 
发 经 验 、 图 书 编撰 经 验 的 积累 。 

编程 词典 系列 软件 以 其 特有 的 学 习 模 式 、 全 面 的 编程 资源 (技术 资源 库 、 视频 资源 库 、 
实例 资源 库 、 项 目 资源 库 、 界 面 资 源 库 、 源 码 资源 库 、 工 具 资源 库 等 )、 高 效 的 查询 与 智 
能 检索 功能 ， 向 读者 诠释 了 一 种 全 新 的 学 习 方 法 。 

无 论 您 是 初学 编程 的 爱好 者 、 大 中 专 院 校 的 师 生 、 初 中 级 程序 设计 人 员 ， 还 是 具有 编 
程 经 验 的 开发 人 员 或 者 从 事 软件 研发 的 专家 ， 使 用 编程 词典 系列 软件 都 可 以 让 您 轻松 上 
手 ， 快 速 编 程 ， 高 效 开发 。 为 了 让 更 多 的 编程 爱好 者 了 解 编程 词典 软件 ， 亲 身体 会 它 给 编 


| 程 带 来 的 便利 ， 下 面 就 来 开发 一 个 在 线 编程 词典 。 


| 12.12 系统 功能 结构 


“在 线 编程 词典 ”是 明日 科技 编程 词典 软件 的 网 页 版 ,实现 了 编程 词典 软件 基本 的 浏 


| 览 功能 ， 其 目的 是 让 浏览 者 通过 网 页 体会 编程 词典 软件 的 海量 数据 资源 给 编程 带 来 的 极 大 
| 便利 。 


在 线 编程 词典 具备 以 下 功能 

回 ”编程 词典 内 容 的 分 类 、 分 块 输出 。 

回 “便捷 的 树 状 导航 菜单 。 

回 ”输出 内 容 的 权限 控制 。 

回 ”体验 内 容 的 合理 输出 。 

在 线 编程 词典 网 站 的 系统 功能 结构 如 图 12.1 所 示 。 


图 12.1 在 线 编程 词典 网 站 的 功能 结构 图 
9 
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12.1.3 文件 夹 架构 


在 线 编程 词典 网 站 的 文件 夹 目录 结构 如 图 12.2 所 示 , 其 中 包括 了 本 模块 中 所 有 的 文件 | 
以 及 文件 夹 。 全 内 


日 名 12 程序 根 目录 | 

田 - 筷 ”1999282828- 默认 输出 文件 
由 加 cs Note 
files 程序 读 取 文件 文件 夫 ! 
四 全 images 背 ! 
田园 js JS 脚本 文件 来 ! 
日 system 系统 文件 来 ! 

由- 国 cache 摆 存 文件 夹 

由 - 国 configs 配置 文件 夫 

田 - 国 lits Smarty 类 库 文件 夹 

BD tplates 模板 文件 夹 


国 accidence htnl 

国 criterion htnl 

国 developer htnl 

国 index .htnl 

国 pda_betton htnl 

国 pdleft htnl 

全 pd_title htnl 

仙 pd_top. htnl 

由 和 toprltte 一 一 一 一 一 一 一 一 一 编译 文件 夫 

国 ”st class ine. php 一 一 一 一 一 一 数据 库 类 的 封装 文件 夫 

国 systen ine.php 一 一 一 一 一 一 一 一 一 类 的 实例 化 文件 夹 

国 systen smarty ine. php 一 一 Smarty 子 置 类 文件 夹 | 
aceidence php ! 
criterion php | 
developer. php 
index, php 
pa_bottom_php 
pd_left. php 
pdnain php 
pd_title. php 
pd_top. php 


图 12.2 在 线 编程 词典 网 站 的 文件 夹 架构 
12.1.4 程序 预览 | 


为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 几 个 页 面 运行 效果 
图 ， 如 果 想 查看 完整 的 效果 图 ， 参 见 配 套 资源 源 程序 。 
编程 词典 在 线 体验 版 入 门 模式 中 “从 零 开始 ”的 运行 效果 如 图 12.3 所 示 。 


辣 凤 同 同 辐 同 网 风 辣 


图 12.3 从 零 开 始 
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| 
编程 词典 在 线 体验 版 开发 模式 中 “技术 中 心 ” 的 运行 效果 如 图 12.4 所 示 。 
| 


EE， 可 J 扫 帮 革 各 芋 ， 二 下 Pt V3 近 从 的 二 认 革 ， 并 出 了 相当 
E21 扬 示 
和 
| ED 
| | E33 返回 一个 双 本 宇 学 点 型 的 信者 主 参 粗 约 正 芝 售 
| 3 ow ET 
| 3 Ea ESTT EECRTIE 
| 1 人 全， 
| I Fs i 
加 到 ET 
| 7 Log 授 回 一 个 观 酉 弃 放 点 玫 的 注 ， 区 证 参 者 的 白 具 对 豆 仙 
3 Se Ei. £2 
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图 12.4 技术 中 心 
编程 词典 在 线 体验 版 应 用 模式 中 “目录 ”的 运行 效果 如 图 12.5 所 示 。 


DateVahae 函 数 
返回 一 个 Vank(Dats) 《日 RE 光 适 》 的 值 ， 


Daicvaetdato) 
内 要 的 4 当知 通 富足 字符 襄 来 达 式 ， 表示 从 100 年 1 月 1 日 到 999 年 2 月 31 日 之 间 的 一 个 日 期 但是， 


如果 和 a 时 一 个 字 行 桌 ， 上 其 由 丙 只 有 就 字 以 及 分 后 失 字 约 日 期 人际 行 ， 则 DateVaine 会 要 可 系统 中 指 宇 的 
的 英文 月 份 名 乏 ， 全 名 或 维 写 均 可 。 例如 , 除了 1230/1991 可 二 3091 之 外 ，DateVaioe 也 识别 December 30| 
DaieVane 饮 安 块 用 由 计算机 系 垢 日 其 设置 的 当前 学 份 ” 


邻 。 如 果 dar* 学 狼 包 售 时 间 信息 ， 刚 DateValve 不 会 近 回 它 但是， 如果 dat 名 省 元 效 时 间 信息 
示 

本 示例 使 用 DateVaie 王 符 宇 和 市 污 换 为 日 HE 蒜泥 - 

DateSu = DaleVabuefFebmnuary 12. 2004] 。。 变量 DateSu 的 回信 为 “2004.02.12 


Te 


| 图 12.5 目录 
12.2 数据 库 设 计 


虽然 PHP 的 最 佳 搭档 是 MySQL 数据 库 ， 但 是 为 了 丰富 读者 数据 库 方面 的 知识 ， 这 里 
使 用 MS SQL Server 2000 作为 本 程序 的 数据 库 。 
12.2.1 创建 数据 库 


| 
| 
| 
| 
| 
| 在 操作 系统 的 任务 栏 中 单 击 “ 开 始 ” 菜 单 ， 选 择 “ 所 有 程序 ”/Microsoft SQL Server/ 


“企业 管理 器 ”命令 ， 启 动 企 业 管理 器 ， 如 图 12.6 所 示 。 


.294 。 
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图 12.6 企业 管理 器 | 


在 图 12.6 所 示 的 窗口 中 ， 右 击 “ 数 据 库 ” 选项， 在 弹出 的 快捷 菜单 中 选择 “新 建 数 据 
库 ” 命 令 ， 打开 “数据 库 属性 ”对 话 框 ， 如 图 12.7 所 示 。 在 该 对 话 框 中 输入 数据 库 的 名 | 
称 ， 然 后 单 击 “ 确 定 ” 按 钮 ， 数 据 库 创建 成 功 。 


后 竹 一 db prooramn 
六 坑 | 天光 件 | 到 日 | 


图 12.7 创建 db_programme 数据 库 
12.2.2 ”创建 数据 表 | 


创建 表 的 过 程 就 是 定义 表 列 的 过 程 。 这 里 在 db_programme 数据 库 中 创建 一 张 名 为 | 
study 的 数据 表 ， 操 作 步骤 如 下 。 | 
(1) 启动 企业 管理 器 ， 展 开 服务 器 组 及 指定 的 服务 器 。 | 
(2) 展开 “数据库” 节点， 展开 指定 的 数据 库 db_programme， 右 击 “ 表 ”选项 ,在 | 
弹出 的 快捷 菜单 中 选择 “新 建 表 ”命令 或 直接 在 工具 栏 中 单 击 米 按钮 。 | 


图 12.8 创建 表 
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。 G) 在 弹出 的 如 图 12.9 所 示 的 窗口 中 ， 输 入 表 的 字段 名 ， 同 时 设置 表 的 数据 类 型 
长 度 和 人 允许 空 等 属性 。 


优 六 提示 : 
~ | 如 果 在 “允许 空 ” 栏 中 设置 了 不 允许 为 空 , 则 插入 或 修改 记录 时 此 选项 不 允许 为 空 值 ， 


否则 会 弹出 错误 提示 . 


图 12.9 编辑 表 字 段 
(4) 插入 或 删除 列 时 ， 可 以 右 击 相 应 的 列 ， 在 弹出 的 快捷 菜单 中 选择 “插入 列 ” 或 


“删除 列 ” 命 令 即 可 。 
(5) 将 ID 设置 为 主键 。 选 中 列 名 为 ID 的 记录 ， 单 击 鼠 标 右键 ， 在 弹出 的 快捷 菜单 


中 选择 “设置 主键 ”命令 ， 或 者 直接 单 击 工具 栏 上 的 里 按钮 ， 即 可 将 该 列 设置 为 主键 ， 如 
”图 12.10 所 示 。 


图 12.10 设置 主键 
| (6) 设置 完成 后 ， 单 击 工具 栏 中 的 “保存 ”按钮 ， 在 弹出 的 如 图 12.11 所 示 对 话 框 中 
| 输入 新 建 表 的 名 称 “study”， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 数据 表 的 创建 。 
| 进 反 人 称 “ey se 

Ws 


Rs 
| DD 


12.11 “选择 名 称 ” 对 话 框 
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说 明 : 


这 里 介绍 的 只 是 在 SQL Server 服 务 器 下 创建 庆 据 库 、 数 据 表 的 方法 ， 并 没有 对 本 程 序 | 
中 所 使 用 的 数据 库 、 数 据 表 进行 详细 的 讲解 。 如 果 要 运行 本 程序 或 者 想 了 解 本 程序 所 使 用 | 
的 数据 库 , 必须 从 配套 资源 中 复制 db_programme 数据 库 , 并 且 要 附加 到 本 机 的 SQL Server | 


2000 数据 库 中 。 | 


12.2.3 分离、 附加 数据 库 


在 SQL Server 2000 数据 库 中 ， 可 以 通过 “分 离 数据 库 ” 来 实现 对 数据 库 文件 的 复制 。 
分 离 后 的 数据 库 文件 在 服务 器 中 不 能 使 用 ， 必 须 重 新 附加 后 才 可 以 使 用 。 


1. 分 离 数据 库 
这 里 以 db_programme 数据 库 为 例 ， 讲 解 分 离 数据 库 的 操作 步骤 。 


(1) 启动 企业 管理 器 ， 展 开 服务 器 组 及 指定 的 服务 器 。 


(2) 展开 “数据 库 ” 节 点 , 右 击 db_programme 数据 库 , 在 弹出 的 快捷 菜单 中 选择 ll 
有 任务 ”/“ 分 离 数据 库 ” 命 令 ， 如 图 12.12 所 示 。 


[Bee | 
3 


Microceh sot err 
Pr Er 


图 12.12 分 离 数 据 库 | 
(3) 在 弹出 的 对 话 框 中 单 击 “ 确 定 ” 按 钮 ， 最 后 弹出 数据 库 分 离 成 功 的 提示 信息 。 | 
2， 附 加 数据 库 | 


附加 数据 库 对 应 分 离 数 据 库 ， 由 它 完成 分 离 后 的 数据 库 文件 的 附加 操作 。 在 本 书 的 配套 资 | 
源 中 提供 的 就 是 可 以 供 读者 附加 的 db_programme 数据 库 文件 。 附 加 数据 库 的 操作 步骤 如 下 : | 
(1) 启动 企业 管理 器 ， 展 开 服务 器 组 及 指定 的 服务 器 。 
(2) 右 击 “ 数 据 库 ”节点 ， 在 弹出 的 快捷 菜单 中 选择 “所 有 任务 ”/“ 附 加 数据 库 ? 
命令 ， 如 图 12.13 所 示 。 | 
(3) 在 弹出 的 对 话 框 中 选择 数据 库 文件 ， 确 定 附 加 的 数据 库 ， 然 后 单 击 “确定 ” 按 | 
钮 ， 即 可 完成 分 离 后 的 数据 库 文件 的 附加 操作 ， 如 图 12.14 所 示 。 | 
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图 12.13 选择 “所 有 任务 ”/“ 附 加 数据 库 ” 命 令 图 12.14 ”附加 数据 库 


首页 的 内 容 应 突出 网 站 主题 及 特色 ， 这 在 网 站 的 生命 周期 中 起 着 非常 重要 的 作用 。 在 
| 线 编程 词典 网 站 首页 应 用 框架 设计 ， 模 拟 编程 词典 软件 的 页 面 布 局 。 


| 123 首页 设计 


12.3.1 首页 概述 


在 线 编程 词典 网 站 首页 的 设计 效果 如 图 12.15 所 示 ， 其 中 包括 网 站 模块 导航 、 功 能 
， 航 条 、 内 容 树 状 导航 、 内 容 展示 区 和 滚动 导航 条 。 


日 国 _m es cg 站 与 得 理 
国 ws 0 的 运行 环境 


[i [|| Os 


图 12.15 在 线 编程 词典 首页 


| 12.3.2 首页 的 实现 过 程 


在 线 编程 词典 网 站 的 首页 由 两 个 文件 组 成 , 即 模板 页 index.html 和 动态 文件 index.php。 
在 index.html 模板 页 中 ， 应 用 框架 技术 设计 网 页 的 布局 。 其 关键 代码 如 下 : 
01 <html> 


02 <head> 
| 03 <meta http-equiv="Content-Type" content="text/html: charset=gb2312"> 
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04 <title>{$conn_type} 编 程 词典 </title> 
05 </head> 

06 <link rel="stylesheet" type="text/css" href="css/link.css"> | 
07 <frameset rows="93,*,25" frameborder="0"> ! 


08 <frame src="pd_top.php?conn type={$conn type}" name="topwindow" scrolling="no"> | 供 站 

09 <frameset cols="200,*"> | 一 一 

10 <frame src="pd leftphp?conn type={$conn type}" name="leftwindow" frameborder="1"> | 

11 <frameset rows="37,*" frameborder="0"> | Note 

12 <frame src="pd title.php" name="titlewindow" scrolling="no"> | 

13 <frame src="pd mainphp" name="mainwindow" scrolling="auto" | 
frameborder="1"> | 

14 </frameset> | 

15 </frameset> | 


16 <frame src="pd_bottom.php" name="bottomwindow" scrolling="no" frameborder="1"> | 
17 </frameset><noframes></noframes> | 


框架 中 的 文件 分 配 如 下 。 

回 “ 框 架 主 体 文件 :index.html、index.php。 

回 “网 站 模块 导航 : pd_top.html、pd_top.php。 
回 ”内 容 树 状 导航 : pd_left.html、pd_left.php。 
回 
回 
回 


功能 导航 条 : pd_title html、pd_title php。 

内 容 展 示 区 : pd_main.html、pd_main.php。 

滚动 导航 条 : pd_bottom.html、pd_bottom php。 | 
在 index.php 文件 中 ， 首 先 包含 Smarty 配置 文件 ， 然 后 创建 模板 变量 ， 最 后 指定 模板 | 


页 。 其 代码 如 下 : 
01 <?php 
02 header("content-type:text/html; charset=gb2312"):; // 设 置 编码 格式 
03 require_once("system/system.inc.php"); // 调 用 配置 文件 
04 iftisset($ GET['conn type']){ 
05 S$conn type=$_GET['conn type']; 
06 jelse{ | 
07 $conn_type=" 从 零 开 始 "; | 
08 } | 
09 $smarty->assign('conn type',$Sconn type); /设置 标题 模板 变量 | 
10 $smarty->display('index.html"): /指定 模板 页 | 
TE | 


12.4 网 站 模块 导航 设计 


网 站 模块 导航 是 编程 词典 软件 功能 模块 的 展示 区 域 ， 它 会 根据 入 门 、 开 发 和 应 用 3 个 
类 别 ， 分 别 展示 出 每 个 类 别 中 包括 的 功能 。 


299's 


pp CE 
12.4.1 ”网 站 模块 导航 概述 


网 站 模块 导航 根据 网 站 的 LOG 中 传递 的 参数 进行 跳 转 , 如 果 LOG 中 传递 的 是 入 门 的 
数据 ， 那 么 网 站 模块 导航 中 将 输出 入 门类 中 的 功能 模块 ， 如 图 12.16 所 示 。 


开发 ”应 用 


中 级 训练 高 级 开发 高 级 训练 


图 12.16 入 门类 中 的 功能 模块 展示 
如 果 LOG 中 传递 的 是 开发 的 数据 ,那么 网 站 模块 导航 中 将 输出 开发 类 中 的 功能 模块 ， 


| 如 图 12.17 所 示 。 


到) 


方案 PO 


[22 中 


办 Pi 图 中 心 


图 12.17 开发 类 中 的 功能 模块 展示 
如 果 LOG 中 传递 的 是 应 用 的 数据 ,那么 网 站 模块 导航 中 将 输出 应 


日 类 中 的 功能 模块 ， 


如 图 12.18 所 示 。 


Am 


由 


遇 码 管理 


图 12.18 ”应 用 类 中 的 功能 模块 展示 
12.4.2 ”网 站 模块 导航 实现 过 程 


责 语 开 册 十 助词 中。 工具 集 霹 理 


网 站 模块 导航 的 实现 , 主要 应 用 到 pd_top.html、pd_top.php, 以 及 js 文件 夹 下 的 flashl.js 


| 和 top.js 文件 。 


pd_top.php 是 动态 PHP 文件 ， 其 主要 功能 就 是 定义 模板 变量 传递 参数 ， 以 及 指定 模板 


| 页。 其 代码 如 下 ; 


01 <?php 

02 session_start(); 

03 require_once 'system/system.inc.php'; 
04 iflisset($_GET['conn type']){ 


// 初 始 化 SESSION 变量 
// 包 含 配置 文件 ， 数 据 库 连 接 、 操 作文 件 


05 S$conn type=$_GET['conn type']; 

06 jelse{ 

07 $conn type=" 从 零 开 始 "; 

0 

09 iflisset($ GET['conn id]){ 

10 S$conn id=$_GET['conn id']: 

11 yelsef 

12 S$conn id="20510111"; 

3 

14 $smarty->assign('conn type',$conn type): // 定 义 模板 变量 传递 参数 
15 $smarty->assign('conn id',$conn id): // 定 义 模板 变量 传递 参数 
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16 $smarty->display('pd top.html’): /指定 模板 页 
17 ?> 
pd_top.html 是 模板 页 。 首先 , 通过 script 标签 调用 js 脚本 文件 夹 下 的 topjjs 和 flashl.js 
文件 。 然 后 ， 定 义 JavaScript 脚本 ， 执 行 Hashl 方法 ， 向 Flash 中 传递 参数 值 。 最 后 ， 根 据 
模板 变量 gconn id 的 值 进 行 判断 ， 进 而 输出 不 同类 别 的 功能 模块 ; 同时 为 输出 的 功能 模块 
创建 超 链接 ， 链 接 到 pd_left.php 文件 ， 同 时 传递 相应 的 参数 。 其 关键 代码 如 下 : 
01 <html> | 


02 <head> 

03 <meta http-equiv="Content-Type" content="text/html; charset=gb2312"> | 

04 ”<title>{Sconn_type} 编 程 词典 </title> | 

05 </head> | 

06 <link rel="stylesheet" type="text/css" href="css/link.css"></link> 

07 @<script type="text/javascript" src="js/top.js"></script> 

08 @<script src="js/flash1.js"></script> 

09 @<body onLoad="MM preloadImages('images/criterion/images002.jpg")"> 

10 <div style="padding-top:inherit; width:100%:; height: 93px: background:url (images/ 
images_013.jpg):"> 

11 <div style="padding-top:inherit; width:171px: height: 93px: float:left; "> 


12 <script type="text/javascript"> 
5 9 flashl('{$conn type}"); 
14 </script> 

15 </div> 


16 <div id="conn" style=" padding-top:0px; width:80%; height: 93px: float:right;"> 
Le © {if$conn id=="20510111' or $conn id—"} 


18 <div id="accidence" style="padding-top:0px: width: 100%: height: 93px; "> 
19 <ul> 
20 © <li style="display:inline: width: 89px;: height: 93px: float: center: text-align:center; 


font-size: 12px"> 

21 <a href="pd left.php?conn id=20510111&table name=study&class type= 从 零 开 始 " 
target="leftwindow"> 

22 <img src="images/accidence/images_01.jpg" border="0" id="Imagel" 

23 ‘onMouseOver="MM_swapImage('Imagel',",'images/accidence/images_001.jpg',1)" | 

24 onMouseOut="MM _swapImgRestore0" /></a></li> | 

25 <li style="display:inline: width: 94px: height: 93px; float: center: text-align:center; | 
font-size: 12px"> 

26 <a href="pd left.php?conn id=20610111&table name=works&class type= 入 门 训练 " 
target="leftwindow"> 

2 <img src="images/accidence/images_02.jpg" border="0" id="Image2" 

28 ‘onMouseOver="MM_swapImage('Image2',",'images/accidence/images_002.jpg',1)" 

29 ‘onMouseOut="MM _swapImgRestore0" /></a></li> 


30 <hul> | 
31 </div> | 
32 Wi | 
33 .<!-- 省 略 了 部 分 代码 --> | 
34 </div> | 
35 </div> | 
36 </body> | 
37 </html> | 
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”的 关键 代码 解析 
| @ topjs: 导入 的 JS 脚本 文件 ， 控 制 超 链 接 图 片 的 样式 。 
@ flashl.js: 向 Flash 中 传递 参数 使 用 的 方法 。 


倚 六 | @@ MM _preloadImages: 通过 onLoad 事件 调用 topjs 中 定义 的 方法 。 
a | @ flash10: 执行 flashl.js 脚本 中 定义 的 方法 ， 并 传递 参数 {$conn type}。 
日 {if $conn id 一 '20510111' or $conn id 一 "}: 通过 Smarty 模板 中 的 让 语句 判断 网 站 模块 属于 哪 一 类 。 


| @ a: 定义 超 链接 ， 链 接 到 pd_leftphp 文件 ， 并 传递 相应 的 参数 值 ， 同 时 应 用 onMouseOver 和 
| onMouseOut 事件 调用 JavaScript 脚本 中 的 方法 ， 控 制 超 链接 图 片 的 样式 。 


12.5 内 容 树 状 导 航 设 计 
在 内 容 树 状 导航 中 ， 通 过 树 状 导航 菜单 来 展示 指定 功能 模块 中 包含 的 内 容 。 
12.5.1 内 容 树 状 导航 概述 


在 内 容 树 状 导航 中 ， 根 据 网 站 模块 导航 中 传递 的 参数 值 进 行 判断 ， 进 而 通过 树 状 导航 
| 菜单 输出 对 应 功能 模块 中 包含 的 内 容 。 
在 内 容 树 状 导航 中 不 仅 要 包含 pd_left.html 和 pd_left.php 两 个 文件 ， 还 要 根据 网 站 模 
块 导航 中 传递 的 参数 值 来 调用 入 门类 (accidence.html、accidence.php )、 开 发 类 
(developer.html、developer.php) 或 者 应 用 类 (criterion.html、criterion.php) 的 内 容 。 
如 果 是 入 门类 , 内 容 树 状 导 航 中 默认 展示 的 是 “从 零 开始 ”功能 模块 的 内 容 , 如 图 12.19 
所 示 。 


开发 ”应 用 


中 级 开发 


四 人 要 开始 


| 日 个 开发 币 委 
| 昌国 忆 训 了解 (视频 ) 


人 | a 2008 明 日 系列 编程 词典 
i 您 的 私人 编程 专家 
量 国 了解 开 发 相关 的 几 介 | 
图 12.19 入 门类 的 内 容 树 状 导 航 
| 如 果 是 开发 类 , 内 容 树 状 导 航 中 默认 展示 的 是 “技术 中 心 ” 功 能 模块 的 内 容 , 如 图 12.20 
所 示 。 


| 
回 技术 中 心 

已 移 函数 开发 参考 
Ty 数 法 于 实 跨 创造 价值 


oem” | 专业 出 版 专业 品质 
图 12.20 开发 类 的 内 容 树 状 导航 


I 。302 。 


如 果 是 应 
所 示 。 
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类 , 内 容 树 状 导航 中 默认 展示 的 是 “源码 管理 ”功能 模块 的 内 容 , 如 图 12.21 


国 大 历史 信息 的 菜单 创造 价值 


是 Few 中 专业 出 版 专业 局 质 
图 12.21 应 用 类 的 内 容 树 状 导航 


12.5.2 内容 树 状 导航 实现 过 程 


在 内 容 树 
件 外 ， 还 包含 


状 导航 概述 中 己 经 了 解 ， 内 容 树 状 导航 除 包 含 pd_left.php 和 pd_lefthtml 文 
入 门 、 开 发 和 应 用 3 类 中 的 6 个 文件 。 其 具体 实现 过 程 如 下 。 


(1) 在 pd_left.php 文件 中 ， 获 取 网 站 模块 导航 的 超 链 接 中 传递 的 参数 值 ， 并 且 将 参 
数值 赋 给 SESSION 变量 ， 然 后 通过 switch 语句 ， 根 据 传递 的 参数 值 进行 判断 ， 进 而 控制 
框架 中 topwindow (网 站 模块 导航 )、leftwindow (内容 树 状 导航 ) 和 mainwindow (内 容 展 
示 区 ) 3 部 分 内 容 的 输出 。 其 关键 代码 如 下 : 


的 值 为 空 


<?php 

session_start(); /初始化 SESSION 变量 
header("content-type:text/html: charset=gb2312"); 

if(!isset($_GET['conn id']) && !isset($_GET["table name])){ /判断 conn id 和 table_name 


$_SESSION['conn id']="20510111"; /设置 初始 值 
$_SESSION['table_ name']="study"; // 设 置 数据 库 名 称 
$_SESSION['class_type']=" 从 零 开始 "; // 设 置 功能 模块 
$conn id="20510111"; 

Jelse{ // 如 果 不 为 空 ， 则 直接 将 它们 的 值 赋 给 SESSION 变量 


$_SESSION['conn id']=$_GET['conn id]; 
$_SESSION['table_ name']=$_GET['table name']; 
$_SESSION['class type']=$_GET['class_type']: 
$conn id=$_GET['conn id']: 


} 

Tequire 'system/system.inc.php'; ”// 包 含 数据 库 连 接 、 操 作 方法 和 Smarty 的 配置 方法 
switch 条 件 控制 语句 ， 判 断 conn_id 的 值 是 否 与 case 关键 字 

中 指定 的 值 相同 ， 如 果 相 同 ， 则 执行 相应 的 操作 


eid 
switch($conn id){ 
case "20510111": /指定 关键 字 
include 'accidence .php': /包含 文件 


S$smarty->assign(phtml',accidence .html); 。 // 将 模板 文件 的 名 称 赋 给 模板 变量 
/通过 Script 脚本 向 指定 的 文件 中 传递 参数 


echo 
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A 
多 %、 
A 


“<script>parent.topwindow.location.hre 人 ='pd_top.php?conn id=20510111';</script>"; 


| 26 echo "<script>parent.mainwindow.location.hre 人 ='pd_main.php?conn id= 
20510111';</script>"; 

27 break; /跳出 循环 

28 case "20610111": 

29 include 'accidence.php'; 

30 $smarty->assign(phtml',accidence.html): 

31 echo "<script>parent.mainwindow.location.href='pd_main.php?conn id= 
”26010111';</script>"; 

32 break:; 

33 …// 省 略 了 部 分 代码 

34 default: 

3 include 'accidence.php': 

36 S$smarty->assign('phtml','accidence.html’); 

Ey echo 

"<script>parent.topwindow.location.href='pd_top.php?conn id=20510111';</script>"; 
38 echo "<script>try {parent.mainwindow.location.href='pd_main.php?conn id= 
20510111';}catch(e){}</script>"; 

39 break: 

40 } 

41 $smarty->display('pd_left.html"); // 指 定 模板 页 

42 ?> 


(2) pd_lefthtml 模板 文件 中 输出 的 内 容 由 switch 语句 中 定义 的 模板 变量 phtml 的 值 
决定 ， 在 pd_left.html 文件 中 应 用 Smarty 中 的 include 函数 包含 由 phtml 指定 的 模板 文件 。 
其 关键 代码 如 下 : 

01 <script language="javascript" sre="js/left.js"></script> 

02 <link rel="stylesheet" type="text/css" href="css/link.css"></link> 

03 <div style="width: 100%:; height: 100%:; padding-left:5Spx; padding-top:Spx:"> 
04 {include file=$phtml} 

05 </div> 


意 : 
在 pd_left.html 模板 文件 中 ， 必 须 载 入 leftjs 脚本 文件 和 link.css 样式 文件 ， 它 们 是 实 
现 树 状 导航 菜单 功能 和 控制 页 面 属 性 的 关键 。 


至 于 入 门 、 开 发 和 应 用 3 类 中 的 6 个 文件 ， 其 实现 原理 是 相同 的 ， 这 里 以 开发 类 中 的 
developerhtml 和 developerphp 为 例 ， 讲 解 其 实现 步骤 。 
(3) 定义 developerphp 文件 ， 获 取 SESSION 变量 中 传递 的 值 ， 执 行 查询 操作 ， 将 查 
询 结果 赋 给 指定 的 模板 变量 ;注册 模板 函数 ， 读 取 指 定 的 文件 。 其 代码 如 下 : 
01 <?php 
02 ilisset($_ SESSION['conn id]) and isset($_ SESSION['table name])){ /判断 SESSION 变 
量 的 值 是 否 为 空 
03 Sconn id="10127105"; // 如 果 SESSION 变量 的 值 为 室 ， 则 直接 为 变量 赋值 
04 S$table_ name="ymzx"; 


有 .304 。 
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$class_type=" 源 码 管理 "; 
}else{ 

$conn id=$_ SESSION['conn id'; 

S$table name=$ SESSION['table name']; 

Sclass type=$_ SESSIONT['class type']; 


$smarty->assign("conn id".Sconn 1d); 
$smarty->assign("class_type",$class_type); 
function unHtml($params) { 
extract($params): 
if(strlen($text)=—12){ 
S$str =is_file("files/". $text.".htm"); 
S$strl =is file("files/".$text.".exe"); 
这 S$SstD{ 
return "files/". $text.".htm"; 
jelse if($strl){ 
return "files/". $text.".exe":; 
}else{ 
return "files/1999.htm"; 


} 
jelse if(strlen($text)==16){ 
S$str =is_file("files/". $text.".htm"); 
这 S$stD{ 
return "files/". $text.".htm"; 
jelse if($str1){ 
return "files/". $text.".exe"; 
}else{ 
return "files/2000.htm"; 
} 
} 


$smarty->register_ function("Util", "unHtml"); 


if($conn id=="10121002@10122105@10123105@10125105"){ 


Sarray_id=explode("@",$conn 1d); 
Sarray_table=explode("@",$table_name); 
for($i=0;$i<count($array_id):$i++){ 

// 执 行 查询 操作 


S$rst[$i] = S$admindb->ExecSQL("select * friom Sarray table[$i] where xxbh like 


'$array_id[$i]%",$conn):; 


43 
44 
45 
46 
47 
48 


} 


Sresult=array_merge($rst[0],$rst[1],$rst[2],$rst[3)):; 


$smarty->assign('subject', $result); 
Jelse{ 
// 执 行 查 询 操作 


Srst = S$admindb->ExecSQL("select * from Stable name where xxbh like 


'$conn id%",$conn); 


49 


$smarty->assign('subject',$rst); 
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/如 果 SESSION 变量 的 值 不 为 空 ， 则 将 SESSION 变量 的 值 赋 给 指定 的 变量 


/设置 模板 变量 

/设置 模板 变量 
/创建 自 定义 函数 

/将 数组 中 的 数据 转换 到 变量 中 
// 判 断 字符 串 的 长 度 是 否 等 于 12 
// 判 断 指 定 的 文件 是 否 存在 

// 判 断 指 定 的 文件 是 否 存在 

// 如 果 文 件 存在 

// 返 回 文件 名 

// 返 回 文件 名 


/和 否则， 返回 一 个 新 的 文件 


// 判 断 字符 串 的 长 度 是 否 等 于 16 
// 判 断 指定 的 文件 是 否 存 在 


/返回 文件 名 
/返回 文件 名 
// 和 否则， 返回 一 个 新 的 文件 


/注册 模板 函数 

// 判 断 变 量 的 值 
// 将 记 值 返回 到 数组 中 

/将 数据 表 名 称 的 值 返回 到 数组 中 
/循环 读 取 数组 中 的 id 值 


/合并 查询 结果 
/将 合并 的 查询 结果 赋 给 模板 变量 


/将 查询 结果 赋 给 模板 变量 


Re 


(4) 创建 developerhtml 模板 文件 ， 通 过 section 循环 语句 ， 读 取 模 板 变量 中 存储 的 数 
据 , 并 且 应 用 直 语 句 进行 判断 , 如 果 指定 字符 串 的 长 度 是 8 字 节 , 则 说 明 是 一 级 导航 菜单 ; 
如 果 指 定 字 符 串 的 长 度 是 12 字 节 ， 并 且 其 前 8 个 字 节 与 一 级 导航 菜单 的 8 个 字 节 相 同 ， 
那么 它 是 二 级 导航 菜单 ， 如 果 指 定 字符 串 的 长 度 是 16 字 节 ， 并 且 其 前 8 个 字 节 与 一 级 导 
航 菜单 的 8 个 字 节 相 同 ， 其 前 12 个 字 节 与 二 级 导航 菜单 的 12 个 字 节 相 同 ， 那 么 它 是 三 级 
导航 菜单 。developerhtml 的 关键 代码 如 下 : 


01 <div style="display:block: width:400px"> 
02 {section name=id loop=$subject} 


03 {1f $subject[id].xxbhlcount_characters—"8"} 

04 <div style="padding-left:6px:; width:400px: height:16px" 

05 ‘onClick="change('div_{$subject[id].xxbh}','icon_div_{$subject[id].xxbh}"):"> 

06 <u> 

07 <li style="display:block: width:16px: height:20px; float:left; font-size:O0px; 


08 background-image:url(images/jian-1.gif)" id="icon_div_{$subject[id].xxbh}"> 
09 <img src='images/jian-2.gif width="16" height="16" /></li> 
10 <li style="display:block: width:16px; height:20px:; float:left; font-size:Opx; text-align:left;"> {Hf 


| Ssubject[id] link!=""} 


11 <img alt="" src="images/developer/02.gif' width="16" height="16"> {else} 
12 <img alt="" src="images/accidence/05.gif"' width="16" height="16"> {/if} &nbsp:</li> 
13 <li style="white-space:nowrap:; display:block: width:345px:; height:20px; 


| padding-top:3px; float:left; 


14 text-align:left; font-size:12px: ">&nbsp; {$subject[id].name}</li> 


15 </ul> 

16 </div> 
| 17 <div id="div_{$subject[id].xxbh}" style=" display:block; padding-left:2px; width:400px; 
height:16px" > 

18 {section name=id12 loop=$subject} 

19 {if $subject[id12].xxbhlcount_characters=="12" && 

$subject[id12].xxbhltruncate:8:""—$subject[id].xxbh } 

20 <div style=" display:block: padding-left:4px:; width:400px; height:16px" 

2 ‘onClick="change('div_{$subject[id12].xxbh}','icon_div_{$subject[id12].xxbh}"):"> 

2 <ul> 

23 <li style="display:block: ”width:16px: height:20px; float:left: 


font-size:Opx; 

24 background-image:url(images/line-down.gif)"></li> 

25 <li style="display:block: ”width:16px: height:20px:; float:left: 
font-size:Opx:; 

26 background-image:url(images/jian-1.gif)" id="icon_div_{$subject[id12].xxbh}"> 

2 <img src='images/jian-1.gif width="16" height="16" /></li> 

28 <li style="display:block: width:16px: height:20px: float:left: text-align:left: font-size:0px:"> {1f 
$subject[id12].link!=""} 

29 <img alt="" src="images/developer/01.gif" width="16" height="16"> {else} 

30 <img alt="" src="images/accidence/06.gif" width="16" height="16"> {/if} </li> 

3 <li onMouseDown="this.style.background=#999999';" 

32 onMouseOut="this.style.background='#FFFFFF" style="white-space:nowrap; display:block: 
width:329px: height:20px; 

33 padding-top:3px: float:left: text-align:left: font-size:12px: ">&nbsp: 
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34 <a href="{Utl text=$subject[id12].xxbh}?conn id={$conn_id}" 
target="mainwindow"> {$subject[id12].name}</a></l> 

cb </u> 

36 </div> 

37 <div id="div_{$subject[id12].xxbh}" style=" display:block:" > 

38 {section name=id16 loop=$subject} 

3 {1f $subject[id16].xxbhlcount characters 一 "16" && 

40 S$subject[id16].xxbhltruncate:8:""==$subject[id].xxbh SS 1 
S$subject[id16].xxbhltruncate:12:""—$subject[id12].xxbh } | 

41 <div style="padding-left:4px; width:400px; height:16px" > | 

42 <u> | 

43 <li style="display:block:; width:16px: height:20px; float:left | 


font-size:O0px:; 
44 background-image:url(images/line-down.gif)"></li> 
45 <li style="display:block: width:16px; height:20px; float:left:; 
font-size:Opx:; 
46 background-image:url(images/line-down.gif)"></li> | 
47 <li style="display:block: width:16px; height:20px; float:left: | 
font-size:0px; | 
48 background-image:url(images/line-center-1.gif)"></li> | 
49 <li style="display:block: width:16px:; height:20px: float:left; text-align:left: | 
font-size:0px:"> | 
50 <img alt="" src="images/accidence/03.gif" width="16" height="16"></li> | 
SY <li onMouseDown="this.style.background='#999999':" | 
52 onMouseOut="this.style.background='#FFFFFF™" style="white-space:nowrap; display:block; 
width:313px; height:20px; 
1 padding-top:3px; float:left: text-align:left; font-size:12px: ">&nbsp; 


54 <a href="{Utl text=$subject[id16].xxbh}?conn id={$conn_id}" 
target="mainwindow">{$subject[id16].name}</a> </li> 

$5 </u> 

56 </div> | 

57 {/f | 

58 {/section} | 

59 </div> | 

60 {/if | 

61 {/section} | 

62 </div> 

63 {ff 

64 {/section} 

65 </div> 


12.6 内容 展示 区 设计 


12.6.1 内容 展 示 区 概述 


在 内 容 展示 区 中 ， 输 出 内 容 树 状 导航 菜单 中 指定 标题 的 完整 内 容 。 如 果 指 定 标题 的 完 
整 内 容 不 存在 ， 则 输出 指定 的 替代 文件 。 其 运行 效果 如 图 12.22 所 示 。 
5 397 


.pp 关 齐 搞 半 精 角 


六 


如 友 旦 由 芮 法、 到达 拓 移 、 环 垢 组成。 我 们 格 从 事 程序 开 安 、 玉 六 的 专业 人 员 ， 称 头 程序 
员 . 在 开发 项目 之 前 都 时 由 软 { 案 没 计 软件 ， 租 序 吕 栋 和 设 计 人 员 的 分 析 总 结 的 开发 方 于 六 纪 
录 发 现 可 本， 测 尖 人 员 全 与 各 厅 员 进行 沟 填 ， 幸 个 未 程 厅 。 当 测试 通 过 

了 宙 东 的 也 果 - 过 程 本 看 出 ， 程 厅 员 完成 的 工作 在 取 个 软件 开发 过 程 中 起 看 关 委 性 的 作用 ， 也 

在 未 ， 缩 写 代码 。 工作 量 也 反 大 的 。 下面 和 一个 将 件 开 发 的 帝 和 可. 


Note 


ne 


tm » 站 


EP 


和 
[BE fsa 
wo ong Ne 
时 PS 


《二 而》 是 要 衬 虱 全 入 不要 二 工具 ， 王 二 贡 的 攻 方 。 内 寄 运 和 技术 ， 项 目 、 方 雪 ， 祖 司 、 视 栅 ， 乔 条 性 击 。 


图 12.22 内容 展 示 区 
12.6.2 内容 展示 区 实现 过 程 


内 容 展示 区 只 包含 一 个 文件 pd_main.php。 在 该 文件 中 判断 超 链接 传递 的 参数 值 ， 并 
根据 该 值 包含 不 同 的 替代 文件 。 其 代码 如 下 : 


| 说明: 
至 此 ， 有 关 在 线 编程 词典 网 站 的 内 容 讲解 完毕 。 这 里 没有 对 功能 导航 条 和 滚动 导航 条 
的 内 容 进 行 详细 讲解 ， 因 为 在 这 两 项 中 没有 具体 的 内 容 。 


| 01 <?php 

| 02 iftisset($_GET['conn id]) && $_GET['conn _ id] 一 "20510111" ){ /判断 超 链接 的 id 值 

| 03 include_once("files/19913.htm"): // 调 用 指定 的 包含 文件 
| 04 jelse if(isset($_GET['conn id]) && $_GET['conn id']=="10121605"){ 

| 05 include_once("files/19913.htm"): // 调 用 指定 的 包含 文件 
| 06 jelse{ 

| 07 include_once("files/2000.htm"); /调用 指定 的 包含 文件 
| 08 } 

| 09 ?> 


| 127 技术 提 疾 


在 在 线 编程 词典 模块 的 开发 过 程 中 ， 多 处 使 用 了 较 新 的 技术 。 为 了 便于 对 该 项 目 各 个 
模块 的 理解 ， 下 面 首先 对 在 线 编程 词典 网 站 中 所 使 用 的 关键 技术 进行 详细 讲解 。 


12.7.1 Smarty 模板 


在 线 编程 词典 网 站 采用 Smarty 模板 实现 网 站 的 动静 分 离 。Smarty 的 版 本 是 


3 。308 。 
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Smarty-2.6.23， 下 载 后 将 libs 文件 夹 存储 于 mr\12\system\ 文 件 夹 下 。 | 
Smarty 的 配置 方法 存储 于 本 项 目的 system 文件 夹 下 ， 名 称 为 system.smarty.inc.php。 | 
配置 文件 的 代码 如 下 : | 


太 
01 <2pbp | 车 
02 require("libs/Smarty.class.php"); // 包 含 模板 文件 | 
03 class SmartyProject extends Smarty{ /定义 类 ， 继 承 模板 类 Note 
04 function SmartyProjectO{ // 定 义 方法 | 
05 S$this->template_dir ="./system/templates/"; 1/ 指定 模板 文件 存储 位 置 | 
06 S$this->compile_dir ="./system/templates_c/":; // 指 定编 译文 件 存 储 位 置 | 
07 Sthis->config dir ="./system/configs/"; // 指 定 配置 文件 存储 位 置 | 
08 S$this->cache_dir = "./system/cache/"; /指定 缓存 文件 存储 位 置 | 
09 } | 
IO | 
11 ?> 


提示 : 
在 包含 Smarty 类 文件 时 , 必须 设置 正确 的 包含 路 径 , 否则 无 法 调用 Smarty 类 文件 (在 
程序 运行 过 程 中 ， 将 出 现 如 图 12.23 所 示 的 错误 信息 )。 


格式 @) 查看 包 帮助 各 


Warning: require(snarty/snarty/Snarty.class.php): failed to 
open strean: No such file or directory in 

F:\PkhPHP \wwu\MR\Instance\23\java\systen\systen.snarty.inc.php 
on line 2 

Fatal error: require(): Failed opening required 
“snarty/snarty/Snarty.class.php" (include_path=".;C:\php5 
\pear*) in F:\PkhPHP\www\HR\InStance\23 
\java\systen\systen.snarty.inc.php on line 2 


图 12.23 ”Smarty 模板 的 路 径 设 置 不 正确 


类 的 实例 化 操作 在 system 文件 夹 下 的 systemiine php 文件 中 实现 ， 返 回 对 象 名 称 为 | 
$smarty。 在 system.inc.php 文件 中 还 包含 了 其 他 类 的 实例 化 操作 ， 其 代码 如 下 : 


01 <?php | 
02 require("system.smarty.inc.php"); // 包 含 Smarty 配置 类 | 
03 require("system.class.inc.php"); // 包 含 数据 库 连 接 和 操作 类 | 
04 $connobj=new ConnDB("mssql"."localhost","sa","","db_programme"):// 数 据 库 连 接 类 实例 化 | 
05 S$conn=$connobj->GetConnId(); // 执 行 连接 操作 ， 返 回 连接 标识 | 
06 $admindb=new AdminDB(: /数据 库 操作 类 实例 化 | 
07 $smarty=new SmartyProjectO: 1/ 调用 Smarty 模板 | 


08 ?> | 

至 此 ，Smarty 模板 的 安装 和 配置 讲解 完毕 。 | 
12.7.2 PDO 数据 库 抽象 层 | 
在 本 项 目 中 ， 通 过 PDO 数据 库 抽象 层 连接 和 操作 SQL Server 数据 库 。PDO 是 与 PHP | 


5.1 一 起 发 行 的 , 默认 包含 在 PHP 5.1 中 。 由 于 PDO 需要 PHP 5 核心 面向 对 象 特性 的 支持 ， 
因此 无 法 在 PHP 5.0 之 前 的 版 本 中 使 用 它 。 
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默认 情况 下 ，PDO 在 PHP 5.2 中 处 于 开启 状态 ， 但 是 要 启用 对 某 个 数据 库 驱 动 程序 的 


支持 ， 仍 需要 进行 相应 的 配置 操作 。 


在 Linux 环境 下 ， 要 使 用 MySQL 数据 库 ， 可 以 在 configure 命令 中 添加 如 下 选项 。 
--with-pdo-mysql=/path/to/mysqlinstallation 
在 Windows 环境 下 ，PDO 在 php.ini 文件 中 进行 配置 ， 如 图 12.24 所 示 。 


人 php_ini - 记事 本 =|D| xx| 
文件 也 编辑 下 ) 格式 0) 查看 W 和 帮助 oD | 
jextension=php_pdo_firebird -dll 


extension=php_pdo_mssq1.d11 
extension=php_pdo_mysq1.d11 


extension=php_pdo_sqlite-dl1 


图 12.24 在 Windows 环境 下 配置 PDO 
要 启用 PDO， 首 先 必须 加 载 “extension=php_pdo.dll”， 要 想 让 其 支持 某 个 具体 的 数据 


库 ， 则 还 要 加 载 对 应 的 数据 库 选项 。 例 如 ， 要 支持 MS SQL Server 数据 库 ， 则 需要 加 载 


“extension=php_pdo_mssql.dll” 选 项 。 


| 注意 : 


在 完成 数据 库 的 加 载 后 ， 要 保存 php.ini 文件 ， 并 且 重 新 启动 Apache 服务 器 ， 修 改 才 


| 能 金 生效 。 


在 本 项 目 中 ， 将 PDO 连接 和 操作 SQL Server 数据 库 的 方法 都 封装 到 类 中 ， 通 过 对 类 


中 方法 的 调用 即 可 完成 数据 库 的 连接 和 管理 操作 。 定 义 的 类 文件 存储 于 本 项 目的 
”MR\12\system 文件 夹 下 ， 名 称 为 system.class.inc.php。 


在 ConnDB 类 中 ， 定 义 连接 数据 库 和 关闭 数据 库 的 方法 。 其 关键 代码 如 下 : 


01 class ConmnDB // 数 据 库 连 接 类 

02 var $dbtype; 

03 var $host; 

04 Var $user; 

05 Var $pwd: 

06 Var $dbname; 

07 function ConnDB($dbtype, $host, $user, Spwd,S$dbname){ // 构 造 方 法 
08 S$this->dbtype=$dbtype; 

09 S$this->host=$host: 

10 S$this->user=$user:; 

11 Sthis->pwd=$pwd; 

i S$this->dbname=$dbname; 

13 } 

14 function GetConnIdO{ // 实 现 数 据 库 的 连接 并 返回 连接 对 象 
15 if($this->dbtype=—"mysql" || $this->dbtype=—"mssql1"){ 

16 Sdsn="$this->dbtype:host=$this->host:dbname=$this->dbname":; 

17 }else{ 

18 S$dsn="S$this->dbtype:dbname=$this->dbname"; 
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19 } | 
20 ty{ | 
21 Sconn = new PDO(Sdsn, Sthis->user $this->pwd); /初始 化 一 个 PDO 对 象 ， 就 是 创建 | 

了 数据 库 连 接 对 象 Spdo | 
22 S$conn->query("set names gb2312"); | 1 
23 return $conn; | 仿生 
24 } catch (PDOException $e) { | 
25 die ("Error!: " . $e->getMessage() . "<br/>"); Note 
26 } | 
27 } | 
28 } | 

在 AdminDB 类 中 ， 定 义 数据 库 的 操作 方法 。 其 关键 代码 如 下 : | 
01 classAdminDB{ // 数 据 库 管理 类 | 
02 function ExecSQL($sqlstrSconn){ | 
03 $sqltype=strtolower(substr(trim($sqlstr),0,6)): | 
04 Srs=$conn->prepare($sqlstr); // 准 备查 询 语句 | 
05 Srs->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 | 
06 if($sqltype=="select"){ | 
07 Sarray=Srs->fetchAll(PDO::FETCH_ASSOC):// 获 取 结果 集中 的 所 有 数据 | 
08 if(count($array)==0 || $rs==false) | 
09 return false: | 
10 else | 
11 return $array; | 
12 }elseif ($sqltype=——"update" || $sqltype 一 "insert" | $sqltype=="delete"){ | 
13 if($rs) | 
14 retum true; | 
15 else | 
16 return false; | 
17 } | 
18 } 

To 


类 的 实例 化 文件 ， 都 存储 于 system 文件 夹 下 的 system.inc.php 文件 中 。 数 据 库 连接 成 | 
功 返 回 的 标识 是 Sconn， 而 数据 库 操作 类 实例 化 的 对 象 是 Sadmindb。 | 


12.7.3 “ 树 状 导航 菜单 设计 | 


为 了 方便 浏览 者 查看 在 线 编程 词典 网 站 的 海量 数据 资源 ， 这 里 通过 树 状 导航 菜单 输出 
资源 列表 ， 其 运行 效果 如 图 12.25 所 示 。 | 


从 零 开始 | 
由 窟 开发 前 奏 

日 罕 第 1 课 初 iRyisual Basic 6.0 
白人 国 Visusl Basic 简 介 ! 
Wisual Basic 的 发 展 
国 misaa Basie 6.0 的 特点 
外国 如 何 学 好 YB 

田 国 VB 6.0 的 安装 与 管理 | 


图 12.25 通过 树 状 导航 菜单 输出 数据 资源 信息 


Sis 


WA 


那么 ， 这 个 树 状 导航 菜单 是 如 何 实现 的 呢 ? 其 开发 的 原理 又 是 怎么 样 的 呢 ? 下 面 就 来 
对 这 些 疑 问 一 一 进行 解答 。 
1. 设计 原理 


树 状 导航 菜单 的 设计 用 一 句 话 就 可 以 概括 : 控制 页 面 中 div 标签 的 展开 和 隐藏 。 这 是 
它 的 根本 之 所 在 ， 虽 然 看 似 简单 ， 但 是 在 具体 的 实施 上 ， 还 需要 应 用 到 一 些 其 他 的 技术 ， 
| 包括 onClick 事件 、JavaScript 脚本 和 div 标签 本 身 的 一 些 属性 。 


2.， 具体 方案 


树 状 导航 菜单 的 具体 开发 步骤 如 下 : 
(1) 创建 div 标签 ， 根 据 从 模板 变量 中 读 取 的 数据 ， 设 置 div 标签 的 id， 设置 div 标 
签 样式 style 中 display 的 值 为 block。 
(2) 在 div 标签 中 通过 onClick 事件 调用 JavaScript 脚本 中 的 方法 ， 并 将 定义 的 div 
标签 的 id 值 作 为 参数 。 
上 述 两 步 操作 都 是 在 Smarty 的 模板 文件 中 完成 的 。 下 面 以 accidence .html 模板 文件 为 
例 ， 给 出 其 具体 代码 。 


01 <div style="display:block; width:400px"> 

02 {section name=id loop=$subject} 

03 {if $subject[id].xxbhlcount_characters—"12"} 

04 @ <divstyle="padding-left:6px:; width:400px: height:16px" 

05 ‘onClick="change('div_{$subject[id].xxbh}','icon_div_{$subject[id].xxbh}");"> 

06 <ul> 

07 @ <li style="display:block: width:16px; height:20px: float:left: font-size:0px; 

08 background-image:url(images/jian-1.gif)" id="icon_div_{$subject[id].xxbh}"> 

09 <img src='images/jian-2.gif width="16" height="16" /></li> 

10 <li style="display:block; width:16px: height:20px; float:left: font-size:0px: text-align:left:"> {Hf 
$subject[id].link!=""} 

11 <img alt="" src="images/accidence/01.gif"' width="16" height="16"> {else} 

2 <img alt="" src="images/accidence/05.gif"' width="16" height="16"> {/if}&nbsp;</l> 

13 <li style="white-space:nowrap; display:block: width:345px; height:20px; 
padding-top:3px; float:left; 

14 text-align:left; font-size:12px: ">&nbsp; {$subject[id].name}</li> 

15 </ul> 

16 </div> 

17 © <divid="div {$subjectfid].xxbh}" style=" display:block; padding-left:2px; width:400px; 
height:16px" > 


18 {section name=id16 loop=$subject} 
19 {1f S$subject[id16].xxbhlcount_characters=="16" && 
S$subject[id16].xxbhltruncate:12:""=—=$subject[id].xxbh } 
20 9 <div style=" display:block: padding-left:4px: width:400px; height:16px" 
| 21 ‘onClick="change('div_{$subject[id16].xxbh}','icon_div_{$subject[id16].xxbh}");"> 
| 22 <ul> 
| 23 <li style="display:block; ”width:16px: height:20px; float:left; 


font-size:Opx; 
24 background-image:url(images/line-down.gif)"></li> 
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25 日 <li style="display:block: width:l6px: height:20px: float:left; 
font-size:Opx:; 

26 background-image:url(images/jian-1.gif)" id="icon div_{$subject[id16].xxbh}"> 

27 <img src='images/jian-1.gif width="16" height="16" /></li> 

28 <li style="display:block: ”width:16px: height:20px; float:left: 
text-align:left; font-size:0px:"> 

29 {if S$subject[idl6].link!=""} <img alt="" src="images/accidence/02.gif' width="16" 
height="16"> {else} 

30 <img alt="" src="images/accidence/06.gif" width="16" height="16"> {/if} </li> 

31 <li onMouseDown="this.style.background='#999999';" | 
onMouseOut="this.style.background='#FFFFFF" | 

32 style="white-space:nowrap; display:block: width:329px: beight:20px; padding-top:3px; 
float:left; text-align:left: 

33 font-size:12px; ">&nbsp; <a href="{Util text=$subject[id16].xxbh}?conn id={$conn id}" 

34 target="mainwindow">{$subject[id16].name}</a></li> 


3 </ul> 
36 </div> | 
37 © <div id="div_{$subject[id16].xxbh}" style=" display:block:" > | 
38 {section name=id20 loop=$subject} | 
39 {1f $subject[id20].xxbhlcount_characters=="20" && | 
40 $subject[id16].xxbhltruncate:12:""==$subject[id].xxbh && | 
$subject[id20].xxbhltruncate:16:""==$subject[id16].xxbh } | 
41 <div style="padding-left:4px: width:400px: height:16px" > | 
42 <ul> | 
43 <li style="display:block; width:16px: height:20px; floatleft | 


font-size:Opx:; | 

44 background-image:url(images/line-down.gif)"></li> 

45 <li style="display:block: width:16px: height:20px: float:left; 
font-size:Opx; 

46 background-image:url(images/line-down.gif)"></li> 

47 <li style="display:block; width:16px: height:20px: float:left; ， 
font-size:O0px:; | 

48 background-image:url(images/line-center-1.gif)"></li> 

49 <li style="display:block: width:16px: height:20px: float:left; text-align:left; | 
font-size:0px:"> 

50 <img alt="" src="images/accidence/03.gif' width="16" height="16"></li> 


51 <li onMouseDown="this.style.background='#999999';" 
52 onMouseOut="this.style.background='#FFFFFF™" style="white-space:nowrap; display:block; 
width:313px; height:20px; 


53 padding-top:3px:; float:left: text-align:left: font-size:12px: ">&nbsp; <a href="{Util 
54 text=$subject[id20].xxbh}?conn id={$conn id}" 
target="mainwindow">{$subject[id20].name}</a> </li> 


55 </nl> 
56 </div> 

57 {if} 

58 {/section} 

59 </div> 

60 {/f} 

61 {/section} 
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Note 


64 
65 
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</div> 
{请 


{/section} 
</div> 


@ onClick 事件 : 调用 change 方法 ， 将 在 @@ 和 四 中 定义 的 id 值 作为 参数 传递 。 
@ id="icon_div_{$subject[id].xxbh}": 设 置 上 i 标签 的 id 值 ,并 设置 其 样式 style 中 的 display 的 值 为 block。 
@ id="div_{Ssubject[id].xxbh}": 设置 div 标签 的 id 值 ,并 设置 其 样式 style 中 的 display 的 值 为 block。 
@ onClick 事件 : 调用 change 方法 ， 将 在 全 和 @ 中 定义 的 id 值 作为 参数 传递 。 
@ id="icon_div_{$subject[id16].xxbh}": 设置 i 标签 的 id 值 ， 并 设置 其 样式 style 中 的 display 的 值 为 


block。 


@ id="div_{$subject[id16].xxbh}": 设 置 div 标签 的 id 值 , 并 设置 其 样式 style 中 的 display 的 值 为 block。 


说 明 : 


在 本 项 目的 入 门 、 开 发 和 应 用 3 个 模块 中 都 用 到 了 树 状 导 航 菜单 功能 ， 这 里 仅 以 入 门 
模块 为 例 对 其 进行 讲解 。 其 中 涉及 到 的 Smarty 模板 技术 已 在 模块 中 进行 详细 讲解 。 这 里 
之 所 以 应 用 了 两 次 onClick 事件 ， 是 因为 实现 的 是 一 个 3 级 导航 的 菜单 。 


(3) 创建 JavaScript 脚本 文件 leftjs， 定 义 change 方法 ， 根 据 传递 的 参数 值 控制 div 


标签 样式 style 
function change(nu,icon){ 


输出 图 片 


Ph display 的 值 。change 方法 的 代码 如 下 : 


// 定 义 change 方法 ， 包 含 两 个 参数 
nu=document.getElementById(nu); // 获 取 参 数 nu 的 了 D 值 
icon=document.getElementById(icon); // 获 取 参 数 icon 的 参数 值 


if(nu.style.display 一 "none"){ // 判 断 ， 如 果 参 数 nu 的 样式 为 none〈 隐 藏 ) 
nu.style.display = "block"': // 则 设置 nu 的 样式 为 block (展开) 
icon.innerHTML = "<img src='images/jian-1.gif width='16' height="16' />"; Uh 
jelse{ 
nu.style.display = "none"; // 如 果 nu 的 样式 为 隐藏 ， 


icon.innerHTML = "<img sre='images/jia-1.gif width 一 16' height="16' >";// 输 出 图 片 


12.7.4 向 Flash 中 传递 参数 
在 本 程序 的 LOG 中 , 应 用 Flash 实现 入 门 、 开 发 和 应 用 3 个 模块 之 间 的 跳 转 ， 并 通过 


Flash 完成 参数 的 传递 ， 其 运行 效果 如 图 12.26 所 示 。 


应 用 


图 12.26 ”通过 Flash 完成 模块 之 间 的 跳 转 
.314 。 
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下 面 一 起 探讨 一 下 如 何 通过 Flash 传递 参数 。 
(1) 创建 JS 脚本 文件 ashl.js， 定 义 向 Flash 中 写 入 的 内 容 。 其 代码 如 下 : 


01 function flashl(m){ 

02 var 了 lash2='<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" 

03 codebase="http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version= 
7,0,19,0" 

04 width="171" height="93">'; 


10 pluginspage="http://www.macromedia.com/go/getflashplayer" 
type="application/x-shockwave-flash" 
11 wmode="transparent"></embed>'; 


05 © flash2+='<param id="swf" name="movie" value="images/bccd.swf?conn type | 
=+mt">'; | 
06 flash2+='<param name="quality" value="high">"; | 
07 flash2+='<param name="allScriptAccess" Value="always"/>'; | 
08 flash2+="<param name="wmode" value="transparent">'; | 
09 flash2+='<embed src="images/bccd.swf' width="171" height="93" quality="high" | 


12 flash2+="</object>"'; 
13 © document.write(flash2); 
72 

< 关键 代码 解析 


@ param id="swf': 在 swf 的 url 中 传递 参数 值 。 
@ document.write: 执行 写 入 操作 。 


(2) 在 Flash 元 件 中 通过 on(release) 方 法 获取 传递 的 参数 值 ， 并 执行 超 链接 。 
@ 通过 this.loaderinfo parameters["conn_ type"] 获 取 传 递 的 参数 值 ; 
var type=this.loaderinfo.parameters["conn type"]: 
在 Flash 编辑 器 中 的 编辑 效果 如 图 12.27 所 示 。 


| var ; 声明 局 部 变量 


变量 (Vy): | type = this.loaderinfo,parameters["conn_type”] 
类 型 0:| 人 包 型 》 。 辣 | 癌 为 类 开源 加 1mport 动作 
号 = 月 日 < 


乓 (release) T 
4 四 


图 12.27 通过 变量 传递 参数 
@ 通过 getURL0O 方 法 执行 超 链 接 。 


getURL("pd_left.php?conn id=20510111&table name=study&class type= 从 零 开 始 &conn type= 
type", "leftwindow", "GET"); 


在 Flash 编辑 器 中 的 编辑 效果 如 图 12.28 所 示 。 


“315。 


AI 


| getuRL : 通知 Web 浏览 器 定位 到 指定 的 URL 
URL(W): | pd_eftphprconn_d-20510111Bkable_name=studybdiass_qype= 从 委 开 始 Bcorn_type=type | 口 和 日 


derinfo. paraneters[” comn_type”]; 


3 行 ; getURL(‘pd jeft.php?conn_d=20510111table_name™studyclass_type= 从 零 开交 conn_type=type”, "eftwndow", "GET"); 
图 12.28 ”通过 getURL 执行 超 链接 
(3) 在 本 程序 使 用 的 FLASH 中 ,创建 了 3 个 元 件 ， 分 别 对 应 入 门 、 开 发 和 应 用 3 个 


模块 。 
元 件 1 的 脚本 代码 如 下 : 


01 on (release) { 


02 Var type=this.loaderinfo.parameters["conn type"]; 
| 03 getURL("pd left.php?conn id=20510111&table name=study&zclass type= 从 零 开 始 
| &conn type=type", 
| 04 "leftwindow", "GET"): 
| 05 } 
| 元 件 2 的 脚本 代码 如 下 : 
| 01 on (release) { 
| 02 var type=this.loaderinfo.parameters["conn_ type"]: 
| 03 getURL("pd_left.php?conn id=10121002&table name=jszx&class type= 技术 中 心 
| &conn type=type", 
| 04 "leftwindow", "GET"); 
| 5 
| 元 件 3 的 脚本 代码 如 下 : 
01 on (release) { 
| 02 var type=this.loaderinfo.parameters["conn_type"]; 
| 03 getURL("pd_left.php?conn id=10127105&table name=ymzx&class type= 源码 管理 
| &conn type=type", 
| 04 "leftwindow", "GET"): 
| 0 


| 12.7.5 Smarty 注册 模板 函数 一 一 读 取 文件 


| 在 本 程序 中 ， 不 仅 使 用 Smarty 模板 实现 网 站 的 动静 分 离 ， 而 且 应 用 了 Smarty 的 注册 
模板 函数 ， 判 断 指定 的 文件 是 否 存在 , 如果 存在 则 返回 文件 名 , 否则 返回 一 个 替代 文件 名 。 
注册 模板 函数 存储 于 accidence.php 中 ， 其 代码 如 下 。 


01 function unHtml($params) { // 创 建 自 定义 函数 

02 extract($params); // 将 数组 中 的 数据 转换 到 变量 中 
| 03 ifl(strlen($text)—16){ // 判 断 当 参数 值 的 长 度 为 16 字 节 时 
| 04 $str is_file("files/".$text.".htm"); 。“”// 判 断 指定 文件 是 否 存在 


和 “ 346“ 
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05 ifSstD{f | 
06 return "files/". $text.".htm"; // 返 回 指定 的 文件 | 

07 jelsef | 

08 return "files/19913 htm": /如 果 指 定 文件 不 存在 ， 则 输出 该 文件 | 

» 4 
10 jelse ifstrlen(Stexb 一 20){ // 判 断 当 参数 值 的 长 度 为 20 字 节 时 |! 窒 一 
11 S$str=is_file("files/".Stext"htm"): /判断 指定 文件 是 否 存在 | 

12 这 SstD{f Note 
13 return "files/". $text.".htm"; // 返 回 指定 的 文件 | 

14 jelse{ | 

15 return "files/2000.htm"; // 如 果 指 定 文件 不 存在 ， 则 输出 该 文件 

16 } | 

17 } | 

mT | 


19 $smarty->register function("Util", "unHtml"); /注册 模板 函数 
在 模板 页 中 ， 调 用 注册 模板 函数 的 方法 如 下 。 


<a href="{Util text=$subject[id20].xxbh}?conn id={$conn id}" target="mainwindow"> | 
{$subject[id20].name}</a> | 


12.7.6 ”FRAME 框架 技术 
在 线 编程 词典 网 站 首页 应 用 框架 设计 ， 共 分 为 5 个 部 分 ， 如 图 12.29 所 示 。 


第 一 部 分 


图 12.29 ”在线 编程 词典 网 站 布局 | 
index.html 框架 页 的 代码 如 下 : | 
01 <html> 
02 <head> | 
03 <meta http-equiv="Content-Type" content="text/html; charset=gb2312"> ! 
04 ”<title>{$conn_type} 编 程 词典 </title> 
05 </head> 
06 <link rel="stylesheet" type="text/css" href—="css/link.css"> | 
07 <frameset rows="93,*,25" frameborder="0"> 
08 <frame src="pd_top.php?conn type={$conn type}" name="topwindow" scrolling="no"> | 


09 <frameset cols="200,*"> | 
10 <frame Src="pd_leftphp?conn type={$conn type}" name="leftwindow" | 
frameborder="1"> | 
11 <frameset rows="37,*" frameborder="0"> | 
12 <frame src="pd title.php" name="titlewindow" scrolling="no"> | 

四 
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框架 主体 文件 : index.html、index.php。 


13 <frame strc="pd_main php" name= mainwindow" scrolling="auto" frameborder="1"> 
14 </frameset> 
| 15 </frameset> 
| 16 <frame src="pd_bottom.php" name="bottomwindow" scrolling="no" frameborder="1"> 
傣 六 | 17 </frameset><noframes></noframes> 
| 框架 中 的 文件 分 配 如 下 。 


回 

加 ”网 站 模块 导航 : pd_top .html、pd_top -php。 

回 ”内 容 树 状 导航 : pd_lefthtml、pd_leftphp。 

回 “ 功 能 导航 条 : pd_title html、pd_title php。 

回 ” 内 容 展示 区 : pd_main html、pd_main.php。 

回 ”滚动 导航 条 : pd_bottom.html、pd_bottom.php。 


12.7.7 ， 超 链接 中 的 图 片 切换 技术 


网 站 模块 导航 中 包含 两 项 技术 : 
回 通过 JS 脚本 向 Flash 中 传递 参数 。 该 技术 已 经 在 前 文 作 了 详细 的 讲解 ， 这 里 不 再 
殉 述 。 


加 ”在 超 链接 中 应 用 onMouseOver 和 onMouseOut 事件 调用 JavaScript 脚本 中 的 方法 ， 
控制 鼠标 滑 过 超 链接 图 片 时 的 样式 。 
超 链接 的 设置 如 下 : 
| 01 <a href="pd left.php?conn id=20510111&ctable name=study&class type= 从 零 开 始 " 
| target="leftwindow"> 
02 <img srTc="images/accidence/images_01.jpg" border="0" id="Imagel" 
03 onMouseOver="MM _swapImage(Imagel',",images/accidence/images_001.jpg',1D)" 
04 onMouseOut="MM_swapImgRestore0" /></a> 
JavaScript 脚本 文件 top.js 中 定义 的 方法 如 下 : 


01 function MM_preloadImages() { //v3.0 


| 02 Var d=document; if(d.images){ if(!d.-MM _p) dMM p=new Array(; 
| 03 Var jj=d.MM_p.length.a=MM _preloadImages.arguments; for(i=0; i<a.length:; i++) 
| 04 if (alil.indexOf("#")!=0){ dMM plil=new Image:; dMM_p[j++].src=a[i]:}} 
| 05 } 
| 06 function MM swapImgRestoreO {//v3.0 
| 07 Var i,x,a=document.MM sr: for(i=0:a&&i<a.length&&(x=af[i])&&x.oSrc:i++) x.src=x.oSre; 
| 0820 
| 09 function MM_ findObj(n, d) { //v4.01 
| 10 Var p,ix; 这 !d) d=document: if((p=n.indexOf("?"))>0&&parent.frames.length) { 
| 11 d=parent.frames[n.substring(p+1)].document; n=n.substring(0,p):} 
| 12 if(!(x=d[n])&&d.all) x=d.all[n]; for (1=0;!x&&i<d.forms.length:i++) x=d.forms[i][n]; 
| 13 for(i=0;!x&&d.layers&&i<d.layers.length:it+) x=MM _ findObj(n,d.layers[i].document); 
| 14 if(!x && dgetElementById) x=d.getElementById(n); return x;: 
os: 
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16 function MM_swapImage() {//v3.0 


到 var 1Tj=0.x,a=MM swapImage.arguments; documentMM sr=new Array; for(i=0;i< | 
(a.length-2):1+=3) | 
18 下 ((x=MM findObj(a[i]))!=nulD{document.MM sr[j++]=x; if(!x.oSrc) x.oSrc=x.sre; | 


X.STc=a[i+2]:} | 食 生 
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12.7.8 Smarty 模板 技术 


在 内 容 树 状 导 航 中 , 系统 根据 网 站 模块 导航 中 传递 的 参数 值 , 控制 着 框架 中 topwindow 
(网 站 模块 导航 )、leftwindow (内容 树 状 导 航 ) 和 mainwindow (内容 展示 区 )3 部 分 内 容 | 
的 输出 。 其 应 用 到 的 技术 如 下 : | 


1.， switch 条 件 控 制 语句 的 充分 应 用 


在 pd_left php 文件 中 ， 通 过 $_GET[] 方 法 获取 网 站 模块 导航 中 参数 conn_id 传递 的 值 ，| 
将 该 值 作为 switch 语句 执行 的 条 件 , 与 case 关键 字 指定 的 值 进行 比较 , 如 果 相 同 则 执行 相 | 
应 的 操作 。 首 先 ， 将 一 个 模板 页 赋 给 一 个 模板 变量 ， 然 后 ， 通 过 script 标签 向 topwindow | 
中 的 pd_top.php〔 网 站 模块 导航 〉 文件 中 传递 参数 ， 最 后 ， 通 过 script 标签 向 mainwindow | 
中 的 pd_main.php《〈 内 容 展示 区 ) 文件 中 传递 参数 。 其 关键 代码 如 下 : | 


01 <?php 
02 并 


03 “switch 条 件 控制 语句 ， 判 断 conn_id 的 值 是 否 与 case 关键 字 
04 ”中 指定 的 值 相同 ， 如 果 相 同 ， 则 执行 相应 的 操作 


05 于 
06 Switch($conn id){ 
07 case "20510111": // 指 定 关键 字 
08 include 'accidence.php'; // 包 含 文件 
09 S$smarty->assign(phtml',accidence.html); 。 // 将 模板 文件 的 名 称 赋 给 模板 变量 
10 // 通 过 script 脚本 向 指定 的 文件 中 传递 参数 
11 echo 
"<script>parent.topwindow.location.href='pd_top.php?conn id=20510111';</script>"; | 
12 echo "<script>parent.mainwindow.location.href='pd_main.php? | 
conn id=20510111':</script>"; | 
13 break: /跳出 循环 


14 “.…// 省 略 了 部 分 代码 
2， 通 过 Smarty 中 的 include0 函 数 在 当前 模板 中 包含 其 他 模板 文件 
在 switch 语句 中 ， 将 指定 的 模板 文件 名 赋 给 一 个 模板 变量 phtml。 此 时 在 内 容 树 状 导 | 
航 的 模板 文件 pd_left.html 中 ， 获 取 模板 变量 phtml 传递 的 值 ， 并 且 将 该 值 作为 Smarty 中 
include0 函 数 包 含 的 模板 文件 。 内 容 树 状 导航 模板 文件 pd_lefthtml 的 关键 代码 如 下 : 


01 <script language="javascript" src="js/left.js"></script> 
02 <link rel="stylesheet" type="text/css" href—"css/link.css"></link> 
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03 <body> 
04 <div style="width: 100%; height: 100%: padding-left:Spx; padding-top:Spx:"> 
ns 和 E 
06 </div> 
</body> 
3. 通过 Smarty 中 的 注册 模板 函数 读 取 指定 文件 
| 这 里 应 用 12.7.5 节 中 讲解 的 Smarty 注册 模板 函数 读 取 指 定 的 文件 。 详 细 内 容 参 见 
| 12.7.5 节 ， 这 里 不 再 袭 述 。 
4. 通过 Smarty 中 的 count characters 和 truncate 方法 统计 和 截取 字符 串 
| 在 入 门类 、 开发 类 和 应 用 类 的 树 状 导航 菜单 中 , 都 应 用 到 Smarty 中 的 count_characters 
| 和 tmuncate 方法 ， 对 指定 字符 串 的 长 度 进行 统计 和 截取 ， 进 而 判断 数据 属于 几 级 菜单 。 
| count_characters 和 truncate 方法 在 本 项 目 中 应 用 的 关键 代码 如 下 : 
| Tr et ay 
上 述 代 码 取 自 developerhtml 模板 文件 。 
5. 通过 树 状 导航 菜单 展示 功能 模块 的 内 容 
有 关 树 状 导航 菜单 的 内 容 参见 12.5.1 节 ， 这 里 不 再 歼 述 。 


12.7.9 文件 检测 技术 


| 内 容 展示 区 中 完整 内 容 的 输出 ， 关 键 是 内 容 树 状 导 航 菜单 中 标题 的 超 链接 值 的 设置 。 
| 这 个 超 链接 的 值 是 从 注册 模板 函数 中 获取 的 。 也 就 是 说 在 注册 模板 函数 中 判断 指定 标题 的 
| 完整 文件 是 否 存 在 ， 如 果 存 在 则 输出 这 个 文件 ， 否 则 输出 指定 的 蔡 代 文件 。 在 本 项 目 中 定 
| 义 了 3 个 注册 模板 函数 ,其 功能 都 是 输出 指定 的 文件 ,这 3 个 文件 分 别 存 储 于 accidence.php、 
| criterion.php 和 developerphp 中 ， 下 面 给 出 在 developerphp 文件 中 具体 的 文件 检测 代码 。 


01 function unHtml($params) { // 创 建 自 定义 函数 

02 extract($params); /将 数组 中 的 数据 转换 到 变量 中 
03 if(strlen($text)—12){ // 判 断 字符 串 的 长 度 是 否 等 于 12 
04 Sstr =is_file("files/". $text.".htm"); // 判 断 指定 的 文件 是 否 存 在 

05 S$strl =is_file("files/".$text.".exe"); 。“”// 判 断 指定 的 文件 是 否 存 在 

06 if($str){ // 如 果 文 件 存在 

07 return "files/".$text.".htm"; /返回 文件 名 

08 }else if($strl){ 

09 return "files/". $text.".exe": // 返 回 文件 名 

10 jelse{ 

而 Teturn "files/1999.htm"; /和 否则， 返回 一 个 新 的 文件 

1 

13 }else if(strlen($text)==16){ /判断 字符 串 的 长 度 是 否 等 于 16 
14 Sstr =is file("files/". $text.".htm"); // 判 断 指 定 的 文件 是 否 存 在 

i if($stn){ 

16 return "files/". $text.".htm"; // 返 回 文件 名 
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17 jelse if($str1){ | 

18 return "files/". $text.".exe"; /1/ 返 回 文件 名 | 

19 yelse{ | 

20 Teturn "files/2000.htm"; // 否 则 ， 返 回 一 个 新 的 文件 | 
21 } | 
29 i 全 二 
23 } l , Note 
24 S$smarty->register function("Util", "unHtml"); // 注 册 模 板 函 数 
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邮件 发 送 系统 模块 


( 铝 # 自学 视频 、 源 程序 : 配套 资源 \》mr\13\) 


电子 邮件 是 Internet 中 应 用 最 广泛 的 服务 之 一 。 通 过 电子 邮件 系统 ， 用 户 
可 以 以 极 低廉 的 价格 轻松 .快捷 地 与 世界 上 任何 一 个 角落 的 网 络 用 户 进行 沟通 、 
交流 。 电子 邮件 的 内 容 多 种 多 样 ， 可 以 是 文字 、 图 像 、 上 声音 等 。 在 构建 企业 站 
点 时 ， 可 以 为 企业 用 户 提供 E-mail 通信 功能 ,从 而 极 大 地 提高 企业 的 工作 效率 ， 
减轻 用 户 的 工作 负担 。 电 子 邮 件 的 收发 技术 是 PHP 程序 员 必 须 事 握 的 技术 之 
一 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


失利 用 mail() 函 数 发 送 电子 邮件 
MH 利用 fsockopen() 阳 数 接收 电子 邮件 
MI 带 附件 邮件 的 发 送 与 下 载 
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13.1 邮件 发 送 系统 概述 


随 着 Tntemet 技术 的 飞速 发 展 ， 越 来 越 多 的 人 选择 网 络 来 进行 即时 沟通 。 本 节 将 要 讲 图 站 
解 的 邮件 发 送 系统 涵盖 了 从 登录 邮件 服务 器 ， 到 发 送 邮 件 、 接 收 邮件 、 查 看 邮件 内 容 以 及 
删除 邮件 等 一 系列 功能 的 实现 方法 ,另外 , 还 详细 介绍 了 WinWebMail 的 安装 和 配置 方法 。， 


13.1.1 电子 邮件 简介 


电子 邮件 (E-mail) 是 mntemet 中 应 用 最 广泛 的 服务 之 一 ， 具 有 快速 、 便 捷 、 低 成 本 等 | 
优势 ， 特 别 适 合 网 络 上 的 信息 交流 。 有 了 电子 邮件 系统 ， 用 户 便 可 以 通过 从 因特网 服务 商 | 
CISP) 那里 申请 的 账号 或 邮箱 进行 信件 的 收发 、 访 问 等 操作 ， 轻 松 、 快 捷 地 实现 与 世界 | 
各 地 的 朋友 进行 沟通 、 交 流 。 

电子 邮件 地 址 的 格式 是 USER@SERVER.COM。 其 中 包括 3 部 分 : 第 1 部 分 USER 代 | 
表 用 户 邮箱 的 账号 ， 对 于 同一 个 邮件 接收 服务 器 来 说 ， 这 个 账号 必须 是 唯一 的 ; 第 2 部 分 | 
@ 是 分 隔 符 ; 第 3 部 分 SERVER.COM 是 用 户 邮 箱 的 邮件 接收 服务 器 域名 ， 用 以 标识 其 所 | 
在 的 位 置 。 | 


13.1.2 ”功能 结构 


本 章 将 要 开发 的 邮件 发 送 系统 的 所 有 信息 都 保存 在 POP3 服务 器 上 ， 所 以 不 涉及 后 台 
管理 及 后 台数 据 库 。 该 系统 的 功能 结构 如 图 13.1 所 示 。 | 


图 13.1 邮件 发 送 系统 功能 结构 图 


13.1.3 ”程序 预览 


为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 几 个 页 面 运行 效果 
“393% 本 


图 ， 如 果 想 查看 完整 的 效果 图 ， 参 见 配套 资源 源 程序 。 
登录 邮件 服务 器 的 页 面 如 图 13.2 所 示 。 


全 由 什 发 这 | 奈 27 de 
Note A 


图 13.2 登录 邮件 服务 器 
邮件 发 送 系 统 中 发 送 带 附件 邮件 的 页 面 运行 效果 如 图 13.3 所 示 。 


13.3 发送 带 附件 的 邮件 
邮件 发 送 系 统 中 收 件 箱 的 运行 效果 如 图 13.4 所 示 。 


发 件 人 时 间 大 小 
inir@sysien aail Tue, 26 Apr 2011 17:18:34 2738D 
nir@systen aail Tue, 26 Apr 2011 17:38:36 205.N 
nir@systen nail Tue, 26 Apr 2011 17:48:39 1830. 
nir@sysien aail Tue, 26 Apr 2011 16:01:45 273D 

pesina con Tue, 26 Apr 2011 18:06:57 2518D 


mir@systen xail Tue, 17 Nar 2011 14:00:30 205.N 
mir@sysien mail Toe, 17 Nar 2011 14:02:20 273D 
mir@sssten xail Tue, 17 Nar 2011 14:08:54 180.N 

各 的 向 乱 中 共有 邮件 : & 浊 点 放空 间 : 311 下 


图 13.4 收 件 箱 的 运行 效果 
邮件 系统 中 查看 邮件 详细 信息 的 页 面 运行 效果 如 图 13.5 所 示 。 


了 。324 。 
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发 件 人 : 
忠仁 人 s 
印 件 主 是: 程序 到 去 
To, 17 Nay 2011 14:04:54 
邮件 80. 7031251 
附件 : ta 三 册 phe see 1 09 所 手册 .btn 
恒 伯 发 天线 ， 出 安 至 有 件 ! 


图 


图 13.5 查看 邮件 的 详细 信息 


13.2 ”邮件 收发 的 原理 


在 具体 讲解 邮件 发 送 系统 功能 之 前 , 先 来 了 解 一 下 邮件 发 送 系统 发 送 邮件 、 接 收 邮件 、 
查找 邮件 及 删除 邮件 实现 的 基本 原理 。 


发 送 


BP 件 实现 相对 简单 ， 用 PHP 自身 提供 的 mail0 函 数 就 可 以 完成 该 功能 。 需 要 注意 


的 是 ， 由 于 mailO 函 数 自身 不 能 进行 SMTP 认证 ， 因 此 需要 安装 一 个 SMTP 服务 器 转发 邮 


件 ， 并 且 需 要 在 php.ini 中 设置 SMTP 服务 器 地 址 及 端口 号 。 


接收 由 


〖 件 需要 利用 fsockopen0 〇 函数 来 实现 。 首 先 利用 该 函数 连接 POP3 服务 器 并 返回 


一 个 Socket 句柄 , 之 后 就 可 以 利用 fputs0 函 数 将 POP 命令 发 送 给 服务 器 , 利用 fgets0 函 数 


接收 POP 


服务 器 反馈 回来 的 信息 《如 邮件 头 、 邮 件 内 容 )， 从 邮件 头 中 就 可 以 提取 发 件 人 


地 址 、 发 件 时 间 等 信息 。 


查找 


人 地 址 ， 轨 


删除 


BP 件 时 ,系统 首 先 遍历 所 有 POP 服务 器 中 的 信件 , 并 提取 所 有 信件 的 主题 和 发 件 
昌 果 发 现 与 用 户 查找 的 主题 或 发 件 人 地 址 相同 的 邮件 ， 就 将 该 邮件 显示 出 来 。 
〖 件 只 需 用 fputs0 函 数 将 POP 命令 DELE 写 入 服务 器 即 可 。 需要 注意 的 是 , 删除 


邮件 必须 由 命令 QUIT 来 执行 ， 否 则 即使 发 送 了 DELE 命令 也 无 法 完成 删除 工作 。 

以 上 内 容 为 整个 邮件 发 送 系统 实现 的 核心 思想 。 在 此 笔者 将 收发 邮件 的 各 项 功能 封装 
到 POP3 类 中 ， 这样 既 符 合 PHP 面向 对 象 的 发 展 趋势 ， 又 为 系统 带 来 了 很 好 的 移植 性 ， 并 
且 为 日 后 的 维护 工作 带 来 很 大 方便 。 


13.3 通过 WinWebMail 配置 邮件 服务 器 


在 本 模块 中 ， 我 们 通过 第 三 方 的 WinWebMail 邮件 服务 器 ， 在 本 地 测试 电子 邮件 系统 
的 功能 。 下 面 介绍 一 下 WinWebMail 的 安装 和 配置 。 
(1) 打开 如 图 13.6 所 示 的 “WinWebMail 3.7.7.1 标准 版 ”安装 界面 , 单 击 Next 按钮 。 


52323” 


vinvebiail 3.7.7.1 标准 版 加 | 


Welcome to WirWebM 相 37.7.1 标 准 版 Setup program This 
pogam wal nstal WWebM 3771 标准 所 on your compuier 


tis shong tecommended thal you ext al Windows programs 
before running ths Setup Progam 


Chck Cancel lo qt Setup and close ang programs you have 
running Chck Ne to coninue wih the Setup program 


WARNING: This program is protected by copyright law and 
temational ieates 


Unauhorized reproduction or distibution of has program. or any 
porton of l, may renulk n severe civi and cirminal penalies, and 
wd be prosecued to the mamum extent possible under law 


[ES 


图 13.6 “WinWebMail 3.7.7.1 标准 版 ”安装 界面 
| (2) 弹出 如 图 13.7 所 示 “WinWebMail 软件 产品 最 终 用 户 许可 协议 ”对 话 框 ， 单 击 
Next 按钮 。 


vnil 3.T7.7. 了 标准 地 
WinWebMall 软 件 产品 最 图 用 户 许可 协议 


WiriWebM 台 软件 产品 最 图 用 户 许可 协议 


A ， 并 爱 著 作 可 法 、 国 际 版 权 公约 以 及 其 它 知识 产权 
法 建 及 条 的 之 保护 。 
本 上 软件 产品 | 奴 系 授权 使 用 。 
4、 授权 
授权 您 仅 在 一 台电 脑 上 安装 使 用 本 『 软 件 产品 】 。 


2 、 限 制 . 
您 应 保留 所 有 『 软 件 产品 捍 贝 上 之 团 作 可 标示 。 


加 


wise Instalation Weard? 


| 

a EE 

图 13.7 “WinWebMail 软件 产品 最 终 用 户 许可 协议 ”对 话 杠 
| 

| 

| 

| 


(3) 弹出 如 图 13.8 所 示 Destination Location 对 话 框 ， 从 中 设置 安装 路 径 ， 然 后 单 击 
| Next 按钮 。 


泥 Winwebiail 3.7.7.1 标准 上 


| Destination Location 


Selup wilinstal WiWebMai 3771 标 准 版 mthe folowing folder 
Toinatalinio a diferen aider chck Browse and select ancther iolder 
You can choose nal to natal WeiWebMal 377.1 标准 版 by cicking Cancal io edt Setup 


Destination Folder 一 一- 


CNWirWebMal Blowse. 


/se Instalation Ward? 


‘Bok [LE |] _ cee 


图 13.8 ”Destination Location 对 话 框 
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(4) 弹出 如 图 13.9 所 示 Select Program Manager Group 对 话 框 ， 单 击 Next 按钮 。 | 


党 Wanweiaail 3.7.7.1 标准 芯 


Select Program Manager Group 
于 4 
Enker the name of the Program Manager group to add WiriWebM 避 377.1 标 准 版 icons to | 傅 生 | 
| 
ee ns Note 


so Newak | 
Mrah De 2 | 
Maceoh 39L Se 1 版本 切换 ! 
Mearsoht SOL Sover 2005 

hosoh Vieual Studio 2009 
[ee 
endSudolaEcpse-511 > 


| 
| 

wee naalaionweady ! 
| 

| 


| ca 
图 13.9 _ Select Program Manager Group 对 话 框 


(5) 弹出 如 图 13.10 所 示 对 话 框 ， 单 击 Finish 按钮 ， 完 成 Winwebmail 3.7.7.1 标准 版 


混 Wianwebaail 3.7.7.1 标准 版 


WinWebM 加 3771 标 准 版 has been successhdy inetaled 


Press the Fnah butlon to esi this netalalion 


= 


图 13.10 完成 WinWebMail 3.7.7.1 标准 版 的 安装 


(6) 启动 WinWebMail， 此 时 可 以 在 其 界面 右 下 角 看 到 一 个 带 红 色 X 的 蓝 色 图 标 , 右 | 
击 该 图 标 ， 在 弹出 的 快捷 菜单 中 选择 “服务 ”命令 ， 如 图 nl 所 示 。 | 


选择 系统 设置 


WinWebMail Server 


选择 域名 管理 


图 13.11 在 快捷 菜单 中 选择 “服务 ”命令 
3327 


WA 


(7) 进入 如 图 13.12 所 示 的 “WinWebMail 服务 ”界面 ， 选 中 “修改 ” 复 选 框 ， 输 入 


DNS 的 包 地 址 ， 单 击 “ 启 动 WinWebMail 服务 程序 ”按钮 ， 然 后 单 击 V (绿色 对 号 ) 按 


钮 ， 完 成 服务 选项 配置 。 


人 当 mhS 查 ix 记录 失败 ， 从 DRS 根 服务 器 查询 | 设置 | 
ns 鲁 冲 保存 时 间 : [7 天 。 二] | 滞 空 所 有 DIS 缠 冲 数 据 
-局 动 设置 
启动 方式 普通 个 快速 
重新 读 取 邮箱 空间 使 用 数据 


一 生 nfebllsil 服 务 程序 a 
9 v x 
图 13.12 WinWebMail 服务 
(8) 重新 进入 如 图 13.11 所 示 界 面 ， 右 击 右 下 角 的 嘱 图标， 在 弹出 的 快捷 菜单 中 和 


择 “ 域 名 管理 ”命令 ， 弹 出 如 图 13.13 所 示 界 面 。 在 该 界面 中 添加 域名 ， 然 后 单 击 v 〈 绿 
| 色 对 号 ) 按钮 。 


(9) 重新 进入 如 图 13.11 所 示 界 面 ， 右 击 右 下 角 的 号 图标 ， 在 弹出 的 快捷 菜单 中 选 


择 “ 系 统 设置 ”命令 ， 在 弹出 的 界面 中 依次 添加 用 户 名 、 密 码 ， 选 择 所 在 域名 ， 然 后 单 击 
| (绿色 对 号 ) 按钮 ， 如 图 13.14 所 示 。 至 此 ，WinWebMail 配置 完毕 。 


所 收 必 规 则 “| 吧 靖 | 鸣 TS? 邮件 接收 自 防护 
[< 


选择 域名 [所 有 成 员 习 [EE 


Deads1f8 systen. nail 系统 管理 员 


证 义 域 名 DX 可 国 
EEC ) | 
[EE 

注释 :| 


厂 禁用 帐号 下) 厂 限制 外 发 。 接 入 方式 : |http/sntp/pop3, inap4 并 
景 后 登录 日 期 : 期 满 日 期 YYYYIMDD); 


ET 添加 9) ET 


外 vw| X% 号 [4 x 


图 13.14 WinWebMail 系统 设置 


图 13.13 WinWebMail 域名 管理 
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13.4 登录 邮件 服务 器 


用 户 登录 邮件 服务 器 时 ， 在 如 图 13.15 所 示 的 登录 界面 中 输入 POP3 服务 器 地 址 、 用 | 依 内 
户 在 POP3 服务 器 注册 的 邮箱 名 称 和 密码 ， 单 击 “ 登 录 ” 按钮 ， 将 数据 提交 到 chkloginphp ， 
文件 中 ， 完 成 与 服务 器 的 连接 。 


由 什 发 送 | 不 7 时 
> 月 疡 委 在 
服务 瑚 : 192.168.1,59 
邮 箱 : [dninBeystennail 


明日 科技 Er 
未] 莉 到 


图 13.15 ”登录 邮件 服务 器 

在 chklogin php 文件 中 ， 获 取 表单 提交 的 服务 器 地 址 、 邮 箱 名 称 和 密码 ， 调 用 POP3 | 

类 中 的 open 和 login 方法 ， 完 成 与 服务 器 的 连接 ， 并 且 将 登录 成 功 后 的 服务 器 地 址 、 邮 箱 
名 称 和 密码 赋 给 指定 的 SESSION 变量 。 其 关键 代码 如 下 : 


01 <?php 

02 session_ start(); 

03 include("mailclass.php"); // 包 含 发 邮件 类 

04 Shostname=$_ POST['servername']; // 获 取 提 交 的 服务 器 地 址 

05 $mailname=substr($ POST['mailname'],0,strpos($_POST['mailname"],'(@")); /获取 提交 的 
用 户 邮 箱 地 址 | 

06 $mailpwd=$ POST['mailpwd']; // 获 取 提 交 的 用 户 邮 箱 密码 

07 Srec=new pop3($hostname,110,10): // 对 发 邮件 类 进行 初始 化 


08 ”if(!Srec->open0){ ”// 如 果 调 用 该 类 的 open 方法 返回 false 值 , 则 说 明 没有 成 功 连接 上 服 
务 器 ， 这 时 给 出 提示 


09 echo"<script>alert( 对 不 起 ， 无 法 连接 服务 器 ! "):history.back();</script>"; 
10 exit:; 
11 


} 
12 /如 果 调 用 发 邮件 类 的 login 方法 返回 false 值 ， 则 说 明 用 户 输入 的 名 称 或 密码 错误 
13 if(!$rec->login($mailname,$mailpwd)){ 
14 echo"<script>alert( 对 不 起 ， 您 的 用 户 名 或 密码 输入 错误 ! '):history.backO;</script>"; 


15 exit; 


} 

17 $_SESSION['host]=$hostname;// 如 果 用 户 登 录 成 功 ， 则 注册 SESSION 变量 host 保存 服务 器 地 
址 以 备 其 他 页 面 使 用 | 

18 ”// 如 果 用 户 登录 成 功 ， 则 注册 SESSION 变量 user 保存 用 户 邮箱 地 址 以 备 其 他 页 面 使 用 

19  $_SESSION['user]=$ POST['mailname']:; | 

20 $_SESSION[pass']=$mailpwd; // 如 果 用 户 登录 成 功 ， 则 注册 SESSION 变量 pass 保存 用 户 邮箱 
密码 以 备 其 他 页 面 使 用 

21 ”Srec->close(); /调用 发 邮件 类 的 close 方法 关闭 与 POP3 服务 器 的 连接 

3 header("location:list.php"); 。” // 定 位 到 发 件 箱 页 面 
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13.5 邮件 收发 


仿生 13.5.1 邮件 收发 模块 概述 


成 功 登 录 邮 件 服务 器 之 后 ， 就 可 以 完成 邮件 的 发 送 、 接 收 、 查 看 、 吾 找 和 删除 了 。 邮 
， 件 发 送 系统 提供 的 功能 如 图 13.16 所 示 。 


图 13.16 ”邮件 发 送 系统 的 功能 
| 在 “发 件 箱 ” 中 可 以 完成 无 附件 邮件 和 带 附件 邮件 的 发 送 操作 ， 在 “ 收 件 箱 ” 中 可 以 
| 查看 接收 到 的 邮件 主题 ， 以 及 邮件 的 具体 内 容 ， 同 时 还 可 以 完成 附件 的 下 载 操 作 ， 在 “ 查 
| 找 邮件 ”中 可 以 根据 邮件 主题 和 发 件 人 对 邮件 进行 查找 ;“ 刷 新 ”功能 用 于 完成 对 页 面 的 
| 刷新 操作 ， 以 此 显示 邮箱 最 新 收 到 的 邮件 信息 ;“ 退 出 ”用 于 实现 当前 登录 用 户 的 退出 。 


13.5.2 发送 邮件 


| 发 送 邮 件 的 操作 由 list.php 和 sendmail.php 两 个 文件 完成 。 在 list.php 中 定义 发 送 邮件 
| 表单， 如 图 13.17 所 示 。 


程序 到 试 ， 发 送 带 附件 的 邮件 


| 13.17 “发送 邮件 


为 了 缓解 服务 器 压力 ， 在 添加 发 送 邮件 的 信息 时 ， 应 用 JavaScript 脚本 对 表单 提交 的 
数据 进行 了 验证 ， 其 中 发 件 人 、 收 件 人 和 邮件 内 容 为 必 填 内 容 ， 并 且 格式 必须 正确 ， 而 附 
， 件 为 可 选项 ， 抄 送 和 密 送 不 必 填写 。 验 证 表单 提交 数据 格式 是 否 正确 使 用 的 是 JavaScript 
， 脚 本 的 自 定 义 函数 chkinput0， 该 函数 的 部 分 代码 如 下 : 
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01 i giantuheceaaeseiioe 

02 <script language="JavaScript"> 

03 function chkinput(form) /定义 chkinput0 函 数 ， 用 来 验证 用 户 输入 的 信息 

04 { 

05 if(form.mailfrom.value=——"") // 判 断 用 户 输入 的 信息 是 否 为 空 

06 { | 

07 /如 果 用 户 输入 的 信息 为 室 ， 则 利用 window 的 alert0 方 法 弹出 一 个 提示 框 ， 告 诉 用 户 某 
些 必 添 的 内 容 没有 填写 

08 alert(" 请 输入 发 件 人 地 址 1"); 

09 form.mailfrom.select(); // 选 择 当 前 窗 体 元 素 

10 Teturn(false); // 返 回 false 值 使 表单 不 能 提交 

11 了 

12 /省 略 了 功能 类 似 的 代码 ， 详 见 配套 资源 


13 /由 于 电子 邮箱 中 一 定 含有 字符 @， 所 以 通过 用 户 输入 的 邮箱 地 址 是 否 含有 字符 @ 就 可 以 
判断 用 户 输入 的 邮箱 地 址 是 否 正确 ， 函 数 indexOf0) 的 作用 是 判断 该 函数 参数 在 某 字 符 串 中 第 一 次 出 现 的 
位 置 ， 如 果 结 果 小 于 0 则 说 明 没 有 查找 到 该 字符 


14 if(form.mailfrom.value.indexOf('(@'")<0) 

15 1 

16 alert(" 发 件 人 E-mail 格式 输入 错误 !"); 

17 form.mailfrom.select(); 

18 return(false); 

19 } 

20 EE 

21 return(true); // 如 果 用 户 输入 的 信息 通过 验证 ， 则 该 函数 返回 trmue， 从 而 使 
表单 能 够 提交 

22 } 


p41 </script> 


在 sendmail.php 文件 中 ， 首 先 获取 表单 提交 的 邮件 内 容 ， 然 后 对 邮件 标题 和 内 容 进 行 
编码 格式 转换 ， 最 后 根据 附件 的 内 容 进 行 判断 ， 如 果 附 件 为 空 ， 那 么 直接 应 用 mail0 函 数 
完成 邮件 的 发 送 ， 如 果 附 件 不 为 空 ， 则 需要 通过 文件 系统 函数 读 取 附 件 的 内 容 ， 对 其 编码 
格式 进行 转换 ， 定 义 到 邮件 体 中 后 ， 应 用 mail0 函 数 完成 邮件 的 发 送 。 其 关键 代码 如 下 : 


01 a ptt er 

02 <?php 

03 include("function.php"); 

04 $mailffom=$_POST[mailfrom']; /获取 发 件 人 地 址 

05 $mailto=$ POST[mailto]; // 获 取 收 件 人 地 址 

06 $mailece=$ POST[mailcc']: 1/ 获取 抄 送 和 地址 

07 $mailbce=$ POST[mailbcc']; // 获 取 密 送 人 地 址 

08 $mailsubject=base64 encode($ POST[mailsubject]); // 获 取 邮 件 主 题 

09 $mailbody= base64 encode($ POST[mailbody']): // 获 取 邮 件 内 容 

10 这 empty($_FILES[mpload file][mame])){ 

11 Sheaders = "From:$mailfrom \rn Ce:$mailce \Pm Bcc:$mailbcc": /定义 邮件 头 

12 Sheaders .= "Reply-To:$mailfrom\r\n"; 

13 Sheaders .= "Content-type: text/html:\r\n charset=iso-8859-1 \r\n"; 

14 if(@mail($mailto，$mailsubject,$mailbody,$headers)){ // 通 过 mail0 函 数 发 送 邮件 ， 并 
给 出 邮件 发 送 结果 

15 echo "<script>alert( 邮 件 发 送 成 功 ! ):historybackO</script>": 
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16 jelsef 

17 echo "<script>alert(' 邮 件 发 送 失 败 ! ):history.backO</script>"; 

18 } 

19 }else{ 

20 Sfile =$_FILES[mpload file]['tmp_name']: /获取 附件 名 称 

21 Sboundary = uniqid(""): // 定 义 分 割 线 

2 $headers = "MIME-Version: 1.0 \n"; 

23 $headers .= "Content-type: multipart/mixed; Boundary= $boundary mn": 
24 Sheaders = "From:$mailfrom\\n"; // 定 义 邮件 头 

ps if($_FILES["upload file][type]){ // 判 断 附件 类 型 

26 SmimeType = $_FILES['upload file']['type']: 

27 }else{ 

28 $mimeType ="application/unknown"; // 获 取 附 件 类 型 

29 } 

30 $fileName = $_FILES["upload file'"]['name']: /获得 附件 名 称 

31 $fp = fopen($file, "r): /打开 文件 

32 Sread = fread($fp, filesize($file)); // 将 附件 读 入 变量 $read 
33 Sread = base64 encode(Sread); // 转 换 base64 编码 
34 Sread = chunk split($read); // 将 长 字符 串 切 成 由 每 行 76 个 字符 组 成 的 小 块 
ES 必定 义 邮 件 体 *#/ 

36 S$body = "--$boundary-- 


37 Content-type: text/plain: charset=iso-8859-1 

38 $mailbody 

39 --$boundary-- 

40 Content-type: $mimeType: name=$fileName 

41 Content-disposition: attachment:; filename=$fileName 
42 Content-transfer-encoding: base64 


43 Sread 

44 --$boundary--"; 

45 if(mail($mailto, Smailsubject,$body,$headers){ /用 mail0 函 数 发 送 邮 件 
46 echo "<script>alert(' 附 件 发 送 成 功 ! ');history.backQ</script>"; 

47 }else{ 

48 echo "<script>alert(' 附 件 发 送 失败 ! '):history.back()</script>"; 

49 } 

50000 
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说 明 : 
| 当 要 发 送 的 邮件 中 带 有 附件 时 ， 首 先 需要 将 邮件 的 MIME 的 Content-type 类 型 定义 成 
”multipart/mixed， 然 后 定义 邮件 内 容 分 割 线 ( 一事 字 符 或 数字 )， 最 后 将 文件 内 容 编写 在 邮 
| 件 主体 中 进行 发 送 。 


| 注意 : 


PHP 的 mailO 函 数 只 能 将 邮件 发 送 到 SMTP 的 邮件 转发 器 中 ， 然 后 通过 转发 器 将 邮件 
发 送出 去 。 
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13.5.3 ”接收 邮件 

接收 邮件 的 核心 是 调用 pop3 类 中 的 open、login、stat 和 listmail 方法 ， 实 现 邮件 的 接 | 
收 操作 。pop3 类 独立 存储 于 mailclass php 中 ， 将 在 13.7 节 中 进行 详细 讲解 。 这 里 将 具体 | 
介绍 系统 是 如 何 调用 pop3 类 实现 接收 邮件 功能 的 。 收 件 箱 页 面 如 图 13.18 所 示 。 | 食 - 


改作 人 
dinleyeten mail Tue, 25 kpr 2011 11:19:3 


Xxxxxxxx 


有 ELITE 
万 的 才 箱 中 共有 疗 御 # 寺 占用 空间 TS 下 


图 13.18 ” 收 件 箱 页 面 
在 receivemail.php 文件 中 ， 首 先 初 始 化 SESSION 变量 ， 判 断 当 前 用 户 是 否 具有 访问 


权限 ; 然后 通过 include 语句 载 入 pop3 类 文件 mailclass.php; 最 后 ， 实 例 化 pop3 类 ， 通 过 
返回 的 对 象 调用 类 中 的 方法 ， 循 环 输出 收 件 箱 中 的 邮件 信息 。 其 关键 代码 如 下 : 
01 dd rp 


02 <?php 
03 session start(); 
04 if(!lisset($_SESSION["user]){ | 


05 echo "<script>alert(' 请 不 要 非法 登录 本 站 ! ):history.backO:</script>"; | 
06 exit; 

07 “} /以 上 代码 通过 判断 SESSION 变量 user 的 值 是 否 为 空 来 验证 用 户 是 否 非 法 登录 

08 include("mailclass.php"); /包含 发 邮件 类 

09 S$rec=new pop3($_SESSION['host],110,10); // 对 发 邮件 类 进行 实例 化 

10 Srec->open(); // 调 用 发 邮件 类 的 open 方法 来 与 POP3 服务 器 建立 连接 


11 ”// 调 用 发 邮件 类 的 login 方法 来 验证 用 户 身 份 
12 Srec-> login(substr($_SESSION['user'].0,strpos($_SESSION['user'],'@")),$_SESSION['pass']); 


13 Srec->stat(); /调用 发 邮件 类 的 stat 方法 来 获得 邮件 总 数 和 邮件 总 字 节 数 

14 Srec->listmail(); /调用 发 邮件 类 的 listmail 方法 获得 每 个 邮件 的 大 小 及 序号 

| 攻 

16 <?php 

17 ”if($rec->messages<=0){ /判断 邮件 记录 数 

18 > 

I <table width="650" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> | 

20 <tr> | 

21 <tdbgcolor="#66CCFF"><div align="center"> 您 的 收 件 箱 内 无 邮件 ! </div></td> | 

22 </tr> | 

23 </table> | 

24 <?php | 

25 J}else{ /如 果 邮 件 存在 ， 则 执行 下 面 的 操作 | 

DO | 
:553 \ 


~ AIO 


SS 
<table width="650" border="0" align="center" cellpadding="0" cellspacing="0"> 
28 <tr> 
29 <td height="40" bgcolor="#66CCFF"> 
30 <table width="650" height="40" border="0" cellpadding="0" cellspacing="1"> 
3 <tr> 
<td height="20" colspan="2" bgcolor="E9E8E8"><div align="center"> 主 题 </div></td> 
33 <td width="136" bgcolor="E9E8E8"><div align="center"> 发 件 人 </div></td> 
34 <td width="183" bgcolor="E9E8E8"><div align="center"> 时 间 </div></td> 
35 <td width="95" bgcolor="E9E8E8"><div align="center"> 大 小 </div></td> 
36 </tr> 
37 ”<?php 
38 for($i=1;$i<=S$rec->messages;$it++){ /循环 输出 邮件 数据 
39 Srec->getmail($2); 
40 ?> 
41 <tr> 
42 <td width="43" height="20" bgcolor="E9E8E8"><div align="center"> 
43 <a href="deletemail.php?del=<?php echo $i:7?>"><img src="images/ 
drop.png" border="0"></a> 
44 </div></td> 
45 <td width="187" bgcolor="E9E8E8"><div align="left"> 
46 <a href="readmail.php?mailid=<?php echo $i:?>"> 
47 <?php 
48 for($j=0:$j<count($rec->head);$j++){ /通过 循环 提取 所 有 保存 在 head 数组 中 
的 邮件 头 的 信息 
49 if(substr($rec->head[$j],0,20)=="Subject: =?GB2312?Q?"){ // 获 取 邮 件 主 题 
| 50 echo 
quoted_printable_decode(substr(htmlspecialchars($rec->head[$j]),20, 
51 strlen(trim(htmlspecialchars($rec->head[$j])))-22))."<br>\n"; // 输 出 邮件 主题 
52 }else if(substr($rec->head[$j],0,7)=="Subject"){ ”// 去 掉 邮 件 主 题 的 提示 
文字 "Subject:" 
3 echo base64 decode(substr(htmlspecialchars($rec->head[$j]),8, 
54 strlen(trim(htmlspecialchars($rec->head[$j])))-8))."<br>\n"; 
55 } 
56 
57 ?> 
58 </a> 
59 </div></td> 
60 <td height="20" bgcolor="E9E8E8"><div align="center"> 
61 <?php 
62 for($j=0:$j<count(Srec->head):$j++){ 
63 if(substr($rec->head[$j].0,4)=—"From"){ 1/ 获取 发 件 人 地 址 
64 // 去 掉 发 件 人 地 址 的 提示 文字 "From:" 
65 echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec-> 
head[$j]))-5)."<br>n"; 
66 > 
67 和 
68 4 
69 </div></td> 
70 <td height="20" bgcolor="E9E8E8"><div align="center"> 
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SI 
71 <?php 
72 for($j=0;$j<count($rec->head):$j++){ /获取 头 文件 数据 


74 echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars(Srec->head 
[ji]D)))-10)."<brm":// 输 出 时 间 

75 

76 } 

ei ?> 

78 </div></td> 

79 <td height="20" bgcolor="E9E8E8"><div align="center"> 

80 <?php 

81 $size=$rec->mail list[$i]['size"]; // 获 取 文 件 大 小 

82 if($size>=1024){ /计算 文件 大 小 ， 以 不 同 单位 进行 换算 输出 | 

83 echo substr(($size/1024),0,4)."M"; 

84 }elseif($size>1024*1024){ 

85 echo substr(($size/(1024*1024)),0,4)."G"; 

86 }elseif($size<1024){ | 

87 echo ($size)."KB"; | 

88 ) | 

89 ?> ! 
90 </div></td> | 


73 if(substr(Srec->head[$j].0.4) 一 "Date"){ /截取 时 间 数 据 | 


91 </tr> 

92 <2php 

93 Srec->head=NULL; 

94 } 

95 ?> 

96 </table> 

97 </td> 

98 </tr> 

99 </table> 

100 <table width="650" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> | 

101 <tr> | 

102 <td width="114"></td> | 

103 ”<td width="536"><div align="right> 您 的 邮箱 中 共有 邮件 :&nbsp;<?php echo Srec-> 
messages;?>&nbsp; 封 占用 空间 : 


104 <?php ! 
105 S$totalsize=$rec->size; /1/ 计 算 邮 件 服务 器 空间 大 小 | 
106 if($totalsize>=1024){ | 
107 echo substr(($totalsize/1024),0,4)."M":; | 
108 。 }elseif($totalsize>1024*1024){ | 
109 echo substr((Stotalsize/(1024*1024)),0,4)."G"; | 
110  }elseif($totalsize<1024){ | 
111 echo ($totalsize)."KB": | 
M2 | 
T1890 | 
114 &nbsp:</div></td> | 
Ts </tr> | 
116 ”</table> | 
117 ”<?php | 

项 
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| 118 S$rec->close(); /关闭 服务 器 
| 人 9 
| 120 ?> 


13.5.4 ”查看 邮件 详细 内 容 


在 收 件 箱 页 面 (receivemail.php 文件 ) 中 , 单 击 邮件 主题 超 链 接 , 将 跳 转 到 readmail.php 
| 页 中 。 在 该 文件 中 将 根据 超 链接 传递 的 邮件 ID 值 ， 输 出 指定 邮件 的 详细 信息 。 其 仍然 是 
调用 pop3 类 中 的 方法 ， 完 成 邮件 详细 信息 的 输出 操作 。 其 关键 代码 如 下 : 


| 01 <?php 

| 02 session start(); // 初 始 化 SESSION 变量 

| 03 ifl!isset($_ SESSION['user]){ // 判 断 用 户 的 权限 

| 04 echo "<script>alert(' 请 不 要 非法 登录 本 站 ! "):history.back(;</script>"; 

| 05 exit; 

| 06 } 

| 07 Smailid=$_GET['mailid"]; // 获 取 超 链接 传递 的 邮件 ID 值 

| 08 include("mailclass.php"): // 载 入 pop3 类 

| 09 Srec=new pop3($_SESSION['host],110,10); // 实 例 化 pop3 类 

| 10 S$rec->open(); 1/ 连接 邮件 服务 器 

| 11 S$rec->login(substr($_SESSION['user'"],0.,strpos($_SESSION['user’],'(@")),$_SESSION['pass'"]); 
| // 用 户 登 录 

| 12 Srec->stat(); // 验 证 用 户 是 否 正确 登录 

| 13 Srec->listmail|; // 读 取 邮 件 

| 14 Srec->getmail($mailid); // 根 据 超 链接 传递 的 ID 值 ， 获 取 指定 邮件 的 内 容 

| US 

| 16 <table width="650" border="0" align="center" cellpadding="0" cellspacing="1"> 

| 17 <tr> 

| 18 <td width="152" height="25" bgcolor="#CCCCCC"><div align="center"> 发 件 
| 人 :<div><htd> 

| 19 <td colspan="2" bgcolor="#E9E8E8"><div align="left"> 

| 20 <?php 

| 21 for($j=0;$j<count(Srec->head);Sj++){ // 获 取 发 件 人 信息 

| 92 if(substr($rec->head[$j],0,4)=="From"){ // 输 出 发 件 人 

| 23 echo substr(htmlspecialchars($rec->head[$j]).5.strlen(trim(htmlspecialchars($rec-> 
head[$j]))-5)."<br>\n"; 

| 24 } 

| 25°00 

| 26 ?> 

| 2 </div></td> 

| 28 </t> 

| 29 <tr> 

| 30 <td height="25" bgcolor="#CCCCCC"><div align="center"> 收 件 人 :</div></td> 

| 31 <td height="20" colspan="2" bgcolor="#E9E8E8"><div align="left"><?php echo $_SESSION 
| [user]; ?></div></td> 

| 32 </tr> 

| 33 <t> 

| 34 <td height="25" bgcolor='"#CCCCCC"><div align="center"> 邮 件 主 题 :</div></td> 
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<td height="20" colspan="2" bgcolor="#E9E8E8"><div align="left"> 


<?php 
for($j=0:$j<count($rec->head):$j++){ /获取 邮件 主题 | 
if(substr($rec->head[$j],0,20)=="Subject: =?GB2312?Q?"){ /对 邮件 主题 进行 截取 | 
echo quoted printable_decode(substr(htmlspecialchars($rec->head[$]j]),20, | 供 站 
strlen(trim(htmlspecialchars($rec->head[$j])))-22))."<br>\n"; // 输 出 邮件 主题 | 人 


jelse if(substr($rec->head[$j],0,7)=="Subject"){ 
echo base64 decode(substr(htmlspecialchars($rec->head[$j]),8, 
strlen(trim(htmlspecialchars($rec->head[$j])))-8))."<br>\n"; | 


} 
} 
?> 
</div></td> 
</tr> 
<tr> 
<td height="25" bgcolor="#CCCCCC"><div align="center"> 发 件 时 间 :</div></td> 
<td height="20" colspan="2" bgcolor="#E9E8E8"><div align="left"> 
<?php 
for($j=0:;$j<count($rec->head):$j++){ // 获 取 发 件 时 间 


if(substr($rec->head[$j],0,4)=="Date"){ // 对 发 件 时 间 数 据 进 行 截取 输出 
echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec-> head 


} 
} 
?> 

</div></td> 
</tr> 
<tr> 
<td height="25" bgcolor="#ccccce"><div align="center"> 邮 件 大 小 :</div></td> 
<td height="20" bgcolor="#E9E8E8"><div align="left"> 

<?php 


$size=S$rec->mail list[$mailid]['size]; /获取 邮件 大 小 

这 $size>=1024){ /对 邮件 大 小 进行 换算 输出 
echo ($size/1024)."M": 

}elseif($size>1024*1024){ 
echo ($size/(1024*1024))."G"; 

}elseif($size<1024){ 
echo ($size)."KB"; | 


} 
iflisset($rec->body[6])){ // 判 断 是 否 存 在 附件 
$file="; | 
$filename=substr(strrchr($rec->body[6],"="),1,200);// 获 取 附 件 名 称 
for($i=0;$i<count($rec->body):$i++){ // 读 取 附 件数 据 
if($1>7 && $i<count($rec->body)-2){ | 
S$file =base64 decode(Srec->body[$i]): 对 附件 进行 编码 转换 | 
} ! 
file_put contents("file/".$filename,$file); // 将 附件 数据 写 入 到 指 
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| 

82 } 

| 83 ?> 

| 84 /div></td> 

| 85 <td height="20" bgcolor="#E9E8E8"><div align="center"> 
| 86 <input type="button" value=" 删 除 " class="buttoncss" 


| 87 onClick="javascript:window.location='deletemail.php?del=<?php echo $mailid:?>&filename=<?php 


| echo $filename:;?>"> 
Note | 88 Jdiv></td> 


| 89 </tr> 

| 90 <?php 

| 91 ifisset(Srec->body[6])){ // 判 断 附 件 是 否 存 在 ， 存 在 则 创建 附件 下 载 超 链接 
| 92 > 

| 93 <tr> 

| 94 <td height="25" bgcolor="#cccccc"><div align="center"> 附 件 :</div></td> 
| 95 <td width="352" height="20" bgcolor="#E9E8E8"><?php echo' 

| 96 <a href="down.php?filename=". $filename."" target="_blank">'.$filename.'</a>'; ?> 
| 97 </td> 

| 98 <td width="142" bgcolor="#E9E8E8">&nbsp;</td> 

| 99 </tr> 

| 100 <?php 

| 101 } 

| 102 ?> 

| 103 <tr> 

| 104 <td height="325" colspan="3" bgcolor="#E9E8E8" valign="top"> 

| 105 ”<?php 

| 106 ifisset(Srec->body[6])){ 

| 107 echo base64 decode($rec->body[3])."<br>"; 

| 108 jelse{f 

| 109 if(substr($rec->body[1],0,1)!="="){ 

| 110 echo base64_decode(Srec->body[1]); 

| 111 }else{ 

| 112 echo quoted_printable_decode(S$rec->body[1]): 

| 113 } 

| 114 } 

| 113 ?72 

| 116 </td> 

| 117 </tr> 


118 </table> 


13.5.5 下载 附 件 


| 在 查看 邮件 详细 内 容 页 面 中 单 击 附件 超 链 接 ， 将 链接 到 down.php 文件 ， 在 该 文件 中 
| 根据 超 链 接 传递 的 附件 名 称 实现 附件 的 下 载 。 其 运行 效果 如 图 13.19 所 示 。 
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在 down php 文件 中 ， 根 据 超 链接 传递 的 附件 名 称 ， 通 过 header0 函 数 完成 附件 的 下 载 | 


操作 。 其 关键 代码 如 下 : 


01 <?php 

02 session start(); // 初 始 化 SESSION 变量 

03 $filename='file/.$ GET['filename']; /获取 附件 在 服务 器 中 的 存储 位 置 

04 if(!isset($_ SESSION['user]){ // 判 断 当 前 用 户 的 访问 权限 

05 echo "<scripf>alert(' 对 不 起 ， 本 站 暂时 停止 该 文件 下 载 !);history.backO:</script>"; 
06 exit; 

07 } 

08 $fp=fopen($filename,"r"); // 打 开 指 定 的 文件 


09 header("Content-type:application/octet-stream"):; 

10 header("Accept-ranges:bytes"); 

11 header("Accept-length:" .filesize($filename)); 

12 header("Content-Disposition:attachment:filename=". $filename); 


13 echo fread($fp,filesize($filename)); // 输 出 文件 
14 felose($fp); 1/ 关闭 文件 
TS 


13.5.6 ”删除 邮件 


在 查看 邮件 详细 内 容 页 面 中 ， 单 击 “ 删 除 ”按钮 ， 将 跳 转 到 deletemail.php 文件 中 ， 
根据 超 链 接 传递 的 邮件 ID 和 附件 名 称 ， 调 用 pop3 类 中 的 dele 方法 完成 邮件 的 删除 操作 。 
如 果 存 在 附件 ， 则 应 用 unlink0 函 数 删 除 存 储 于 服务 器 下 的 附件 。deletemail.php 文件 的 代 
码 如 下 : 


01 Co eB eb he oka hohe hdd 
02 <?php 

03 session start(); 

04 iflisset($ SESSION['user]){ // 判 断 当前 用 户 的 权限 

05 include("mailclass.php"); // 包 含 pop3 发 邮件 类 

06 Srec=new pop3($_SESSION['host"],110,10): // 对 类 进行 初始 化 
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07 Srec->open(); 1/ 连接 POP3 服务 器 

08 /对 当前 用 户 进行 身份 验证 

09 Srec->login(substr($_SESSION['user'].0.strpos($_SESSION['user],@")),$_SESSION [pass]); 

10 Srec->dele($_GET['del]); /调用 pop3 类 的 dele 方法 来 删除 $_GET[del] 指 定 的 邮件 

Wh Srec->close(): // 关 闭 与 POP3 服务 器 的 连接 ， 并 执行 删除 操作 

12 if(!empty($_GET['filename'])) { 

13 unlink('file/.$_GETT['filename']):; 

14 } 

15 echo "<script>alert(' 删 除 成 功 !");window.location.href='receivemail.php';</script>"; 

16 jelse{ 

ne echo "<script>alert(' 您 不 具备 删除 权限 ， 请 正确 登录 !);window.location.hre 人 = 
‘index.php';;</script>"; 

2 

19 ?> 


13.5.7 ”查找 邮件 


为 了 方便 用 户 ， 该 邮件 发 送 系统 提供 了 两 种 搜索 方式 ， 分 别 为 按 邮 件 主题 进行 查找 和 
按 发 件 人 地 址 进行 查找 。 查 找 邮 件 的 操作 在 findmail.php 文件 中 完成 ， 页 面 运行 效果 如 
图 13.20 所 示 。 


图 13.20 查找 邮件 
查找 邮件 创建 的 form 表单 如 下 : 


01 <form name="forml" method="post" action="<?php echo $PHP_SELF:;?>" onSubmit="return 


chkinput(this)"> 


02 <tr> 

03 <td width="143"><div align="right"> 查 找 邮 件 :</div></td> 

04 <td width="92"><div align="center"> 

05 <select name="findtype" class="inputcss"> 

06 <option selected value=1> 邮 件 主题 </option> 

07 <option value=0> 发 件 人 </option> 

08 </select> 

09 </div></td> 

10 <td width="191"><div align="center"> 

11 <input type="text" name="info" size="30" class="inputcss" style= 
"background-color:#e8f4 人 ff " 


12 onMouseOver="this.style.backgroundColor=#fffffP" 
‘onMouseOut="this.style.backgroundColor='#e8f4ff"> 
</div></td> 
<td width="224"><div align="left"> 
&nbsp;&nbsp;<input type="submit" value=" 开 始 " class="buttoncss" name= 


</div></td> 
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17 </tr> 
18 </form> 
查找 邮件 的 处 理 在 findmail.php 文件 中 完成 ， 其 关键 代码 如 下 。 | 
01 he ee it dk: ce es hy th dd | 食 扩 | 
02 1 
03 -<?php 
04 ifisset($_POST['submit])){ // 判 断 提交 按钮 值 是 否 被 设置 | CE 
05 ?> | 
06 <t> | 
07 <td width="188" height="20" bgcolor="#E9E8E8"><div align="center"> 主 题 </div> </td> | 
08 <td width="153" height="20" bgcolor="#E9E8E8"><div align="center'> 发 件 人 <Jdiv> </td> | 
09 <td width="209" height="20" bgcolor="#E9E8E8"><div align="center"> 时 间 </div> </td> | 
10 <td width="97" height="20" bgcolor="#E9E8E8"><div align="center"> 大 小 </div> </td> 
11 </tr> 
12 <?php 
13 include("mailclass.php"); // 载 入 类 文件 
14 Srec=new pop3($_SESSION['host],110,10); // 实 例 化 类 
15 Srec->open(); /连接 服务 器 


16 $rec->login(substr($_ SESSION['user],0.stpos($_ SESSION[user],@)),$_SESSION['pass']); 


17 grec->stat(; /调用 pop3 类 的 stat 方法 取得 总 邮件 数 和 总 邮件 大 小 | 

18 Srec->listmail0: // 调 用 pop3 类 的 listmail 方法 取得 每 个 邮件 的 大 小 及 序号 | 

19 if($_POST['findtype]==1){ /判断 查询 的 类 型 ，1 表示 根据 邮件 主题 查询 | 

20 $sum=0; // 用 $sum 记录 查找 到 邮件 的 总 个 数 

21 for($i=1:;$i<=$rec->messages:$i++){ // 通 过 循环 遍历 所 有 邮件 

22 Srec->getmail($i); // 取 得 第 $i 封 邮 件 头 及 邮件 内 容 

23 for($j=0;$j<count($rec->head);$j++){ /通过 循环 获取 邮件 头 所 有 内 容 

24 if(substr($rec->head[$j],0,7)=="Subject"){// 获 取 邮 件 主 题 

25 /去 掉 邮 件 主题 的 提示 文字 "Subject:" 

26 S$head1=substr(htmlspecialchars($rec->head[$j]),8,strlen(trim(htmlspecialchars($rec->head 
[$7D)))-8); 

27: } | 

28 } | 

29 if(preg_match("/$_POST[info]/i",trim($head1))){// 利 用 正则 表达 式 进 行 匹 配 查 找 | 

30 S$sum++; /如 果 查 找到 匹配 的 主题 ， 则 使 sam 值 加 1 | 

31 Ee | 

32 <tr> | 

33 <td height="20" bgcolor="#E9E8E8"><div align="left"> | 

34 <a href="readmail.php?mailid=<?php echo $i:?>"> | 

35 <?php | 

36 for($j=0:;$j<count($rec->head):$j++){ /显示 邮件 主题 | 

37 if(substr($rec->head[$j].0.7)=="Subject"){ | 


38 echo substr(htmlspecialchars($rec->head[$j]).8,strlen(trim(htmlspecialchars($rec->head[$j]))) | 
-8)."<br>\n"; | 


39 } | 
40 } | 
41 人 > | 
42 </a> | 
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-5)."<br>m"; 


-10)."<br>mn: 


| 时 给 出 提示 


的 邮件 !</div></td> 


43 </div></td> 

44 <td height="20" bgcolor="#E9E8E8"><div align="center"> 
45 <?php 

46 for($j=0;$j<count($rec->head);$j++){ ”// 显 示 发 件 人 地 址 

47 if(substr($rec->head[$j],0,4)=—"From"){ 

48 echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec->head[$j]))) 
49 站 

50 国 有 

51 ?> 

52 </div></td> 

oF) <td height="20" bgcolor="#E9E8E8"><div align="center"> 
54 <?php 

55 for($j=0;$j<count($rec->head):$j++){ ”// 显 示 发 件 时 间 

56 if(substr($rec->head[$j],0,4)=—="Date"){ 

57 echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec->head[$j]))) 
58 

59 } 

60 ?> 

61 </div></td> 

62 <td height="20" bgcolor="#E9E8E8"><div align="center"> 
63 <?php 

64 $size=8rec->mail list[$il['size]; /显示 邮件 大 小 
65 if($size>=1024){ 

66 echo substr(($size/1024),0,4)."M"; 

67 }elseif($size>1024*1024){ 

68 echo substr(($size/(1024*1024)).0,4)."G": 

69 }elseif($size<1024){ 

70 echo ($size)."KB"; 

71 } 

72 ?> 

VE] </div></td> 

74 </tr> 

75 <?php 

76 } 

Ek Srec->head=NULL: 

78 } 

79 这 $sum==0){ /如 果 $sum 的 值 最 终 还 为 0， 则 说 明 没有 查找 到 匹配 的 邮件 主题 , 这 
80 ?> 

81 <tr> 

82 <td height="20" colspan="4" bgcolor="#E9E8E8"><div align="center"> 对 不 起 ,没有 查找 到 您 要 找 
83 </tr> 

84 <?php 

85 } 


86 ”}elseif($_POST['findtype']=0){ // 如 果 $_POST['findtype"] 的 值 为 0， 则 按 发 件 人 地 址 进行 查找 
87 $sum=0: 
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88 for($i=1:;$1<=$rec->messages; $11+){ | 
89 Srec->getmail($i):; | 
90 for($j=0:9j<count(Srec->head):Sj++){f | 
91 if(substr($rec->head[$j],0,4)—"From"){ | 
9 Shead1=substr(htmlspecialchars($rec->head[$]]),5,strlen(trim(htmlspecialchars($rec->head | 
[87])))-5); | 
93 } 
94 4 | 
95 if(trim($head1)==trim($ POST[infoTD){ | 
96 S$sumt+t+; | 
1 2 | 
98 <tr> | 


99 <td height="20" bgcolor="#E9E8E8"><div align="left"><a href="readmail.php?mailid=<?php 
echo $1:7>"> 


100 ”<?php 
101 for($j=0;$j<count($rec->head):$j++){ 
102 if(substr($rec->head[$j],0,7)=="Subject"){ 


103 echo substr(htmlspecialchars($rec->head[$j]),8,strlen(trim(htmlspecialchars($rec->head[$j]))) 
-8)."<br>\n"; 


104 } 
1050 | 
106 ?> | 
107 </a></div></td> | 
108 <td height="20" bgcolor="#E9E8E8"><div align="center"> | 
109 ”<?php | 
110 。 for($j=0:$j<count(Srec->head):Sj++){ | 
111 if(substr($rec->head[$j],0,4)=—"From"){ | 
112 echo substr(htmlspecialchars($rec->head[$j]),5,strlen(trim(htmlspecialchars($rec->head[$j]))) | 
-5)."<br>\n"; | 
113 } | 
114 } | 
Ts | 
116 </div></td> | 
117 <td height="20" bgcolor="#E9E8E8"><div align="center"> | 
118 ”<?php | 
119 for($j=0:;$j<count($rec->head);$j++){ | 
120 这 substr($rec->head[$j].0.4) 一 "Date){ 


121 echo substr(htmlspecialchars($rec->head[$j]).S.strlen(trim(htmlspecialchars(Srec->head[$j]))) | 
-10)."<br>\n"; 


122 } | 
1233 | 
124 ?> | 
125 </div></td> | 
126 <td height="2" bgcolor="#E9E8E8"><div align="center"> | 
127 <?php | 
128 $size=$rec->mail list[Si]['size']: | 
129 if($size>=1024){ | 
130 echo substr(($size/1024),0,4)."M"; | 
131 }elseif($size>1024*1024){ | 
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136 ?> 


AI 


echo substr(($size/(1024*1024)),0,4)."G"; 
}elseif($size<1024){ 
echo ($size)."KB":; 


141 Srec->head=NULL:; 


142 } 


143 if($sum—0){ 


144 ?> 
145 <tr> 


146 <td height="20" colspan="4" bgcolor="#E9E8E8"><div align="center"> 对 不 起 ,没有 查找 到 您 要 找 的 


邮件 !</div></td> 
147 </tr> 
148 <?php 
149 } 
SOD 
151 } 
52 


13.6 注销 系统 


为 了 保证 个 人 隐私 ， 用 户 使 用 完 本 系统 后 应 及 时 退出 。 本 邮件 发 送 系统 完成 注销 系统 
功能 的 文件 为 lbgoutphp， 其 详细 代码 如 下 : 


01 a pd dd te hh be 
02 <?php 

03 session start(); // 初 始 化 SESSION 变量 

04 session _destroy0; /撤消 所 有 SESSION 变量 

05 header("location:index.php"); // 重 新 定位 到 登录 页 面 


06 ?> 


注销 系统 模块 通过 session_destroy0 函 数 撤消 所 有 的 SESSION 变量 ， 这 样 保 存在 


SESSION 变量 中 的 所 有 与 上 


户 有 关 的 信息 也 将 不 复 存在 。 


13.7 技术 提炼 


13.7.1 利用 mail() 函 数 发 送 电子 邮件 


PHP 中 提供 了 可 以 直接 发 送 电子 邮件 的 mail0 函 数 。 对 于 非 商业 用 的 邮件 系统 使 用 该 
函数 最 为 方便 。 由 于 mail0 函 数 不 能 进行 SMTP 认证 ， 所 以 最 好 安装 一 个 SMTP 服务 器 来 
转发 邮件 。mail0 函 数 的 语法 如 下 : 
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bool mail(string to, string subject string message[ ,string additional headers][ ,string additional parameters]) 
参数 说 明 如 表 13.1 所 示 。 


表 13.1 el lll id deedl 


收 件 人 地 址 。 其 表示 形式 有 多 种 : user@example.com; user@example.com, 
anotheruser@examplecom ; user <user(@example.com>; user <user(@example.com>, 
another user <anotheruser@example.com> 

电子 邮件 的 主题 

电子 邮件 的 内 容 

可 选 参数 ， 用 来 将 一 些 信 息 插 入 到 E-mail 的 头 部 ， 如 发 送 人 地 址 、 抄 送 地 址 、 
密 送 地 址 等 。 该 参数 如 果 包 含 多 项 信息 ， 各 信息 之 间 用 “\rn” 分 隔 
其 他 邮件 参数 


additional headers 


additional parameters 


注意 : 
使 用 mail0) 函 数 发 送 邮件 ， 需 要 在 php ini 中 对 SMTP 邮件 服务 器 的 地 址 和 端口 进行 设 
置 ， 和 否则 邮件 无 法 通过 SMTP 服务 器 发 送出 去 。 设置 如 下 : 
[mail function] 
; For Win32 only. 
SMTP= 192.168.1.42 
smtp_port = 25 


其 中 ，SMTP 是 发 送 邮 件 的 服务 器 地 址 ; smtp_port 为 SMTP 服务 器 的 端口 号 ,一 般 默 | 
认为 25。 | 

邮件 发 送 系统 是 通过 mail0 函 数 将 电子 邮件 发 送出 去 的 ， 所 以 应 该 在 系统 中 架设 
SMTP 服务 器 。 该 邮件 发 送 系 统 利用 mail0 函 数 发 送 邮 件 的 过 程 如 下 ; 


if(@mail($mailto, $mailsubject,$mailbody, S$headers)) { // 通 过 mail0 函 数 发 送 邮件 , 并 给 出 
邮件 发 送 结果 
echo "<script>alert( 邮 件 发 送 成 功 ! ):historybackO</script>": | 
人 "<script>alert(' 邮 件 未 发 送 成 功 ! ):history.backO</script>"; | 
} 
参数 说 明 如 下 。 | 
加 ”$mailto: 收 件 人 地 址 。 
加 ”$mailsubject: 邮件 主题 。 
加 ”$mailbody: 邮件 内 容 。 | 
加 ”S$headers: 邮件 头 内 容 。 | 
如 果 该 邮件 发 送 成 功 或 失败 系统 将 给 出 提示 ， 并 且 用 “@” 屏 蔽 错误 。 | 


13.7.2 ” 带 附 件 的 邮件 发 送 


发 送 带 附件 的 邮件 并 不 是 简单 地 将 附件 内 容 上 传 到 POP3 服务 器 的 某 个 目录 下 ， 而 是 | 
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首先 将 附件 进行 二 进 制 编码 ， 然 后 将 该 编码 连接 到 邮件 内 容 之 后 。 当 然 ， 邮 件 内 容 和 附件 
| 编码 之 间 需 要 用 标记 进行 分 割 ， 该 标记 在 邮件 头 中 定义 。 带 附件 的 邮件 组 织 形式 如 下 : 


01 Received: from mrkj ([192.168.1.227]) by MRZTS with Microsoft SMTPSVC(6.0.3790.0); 
02 Mon, 18 Dec 2006 09:02:40 +0800 

03 Date: Mon, 18 Dec 2006 00:57:26 +0000 

04 Subject: 12 

05 To: zts***(@163.com 

06 Content-type: multipart/mixed; Boundary= 4585e7760018e 

07 From:zts***(@163.com 

08 Return-Path: zts***(@163.com 

09 Message-ID: 

10 X-OriginalArrivalTime: 18 Dec 2006 01:02:40.0187 (UTC) FILETIME=[372680B0: 01C72240] 
11 --4585e7760018 

12 Content-type: text/plain; charset=iso-8859-1 

13 Content-transfer-encoding: 8bit 

14 “(此 处 为 邮件 内 容 ) 

15 --4585e7760018 

16 Content-type: text/plain: name=shop.txt 

17 Content-disposition: attachment: filename=shop.txt 

18 Content-transfer-encoding: base64 

19 ”wffV4rXEtcS1xLXEtcS1xLTvtb21xNK7ICC13LXctcS1xA 一 (此 处 为 附件 的 base64 编码 ) 
20 --4585e7760018-- 

21 


带 附件 的 邮件 的 不 同 部 分 之 间 用 分 割 线 来 分 隔 ， 分 割 线 在 Content--type 头 中 定义 。 例 


| 如 ,“Boundary= 4585e7760018e”， 其 中 4585e7760018e 表示 不 同 部 分 之 间 的 分 割 线 。 邮 件 
| 的 每 个 新 部 分 以 两 个 连 字号 〈--) 和 分 割 线 开始 ; 最 后 一 条 分 割 线 后 也 有 两 个 连 字号 ， 表 
| 示 这 个 邮件 中 没有 其 他 部 分 了 。 在 每 条 分 割 线 后 有 一 些 行 ， 用 来 告诉 邮件 程序 这 部 分 内 容 


内 类 型 。 例 如 ， 上 面 例子 中 第 一 条 分 界线 后 面 以 Content-type: text/plain 开头 的 行 ， 这 些 行 


| 说 明 后 面 的 部 分 是 ISO-8859-1 字符 集 的 纯 文本 形式 ; 跟 在 第 二 条 分 割 线 后 的 行 告诉 邮件 程 
| 序 现在 的 部 分 是 一 个 TXT 文件 ， 其 名 称 是 shop.txt。 Content-disposition 这 行 告诉 邮件 程 
序 如 果 可 能 就 以 内 嵌 的 方式 显示 附件 ， 这 样 新 的 邮件 程序 会 在 消息 后 显示 TXT 文件 的 内 
| 容 ， 如 果 Content- disposition 被 设置 为 attachment， 那 么 邮件 程序 就 不 会 显示 TXT 文件 的 
| 内 容 , 而 是 显示 一 个 连接 到 文件 的 图 标 , 收 件 人 如 要 查看 附件 的 内 容 , 必须 单 击 这 个 图 标 。 
| 一般 情况 下 ， 如 果 附 件 是 一 些 文本 ， 可 将 Content-disposition 设置 为 nline， 这 是 因为 现在 
| 大 部 分 邮件 程序 能 够 不 借助 其 他 浏览 器 而 直接 显示 附件 (文本 ) 的 内 容 。 如 果 附件 不 是 文 
| 本 《比如 图 片 或 其 他 类 似 的 内 容 )， 则 Content-disposition 应 设置 为 attachment。 


13.7.3 ”利用 fsockopen() 函 数 接收 电子 邮件 


上 面 介绍 的 mail0 函 数 只 能 发 送 邮 件 而 无 法 接收 ， 而 全 ockopen(O) 函 数 除了 可 以 发 送 邮 


， 件 外 ， 还 可 以 接收 邮件 。fsockopen0 函 数 的 语法 格式 如 下 : 


int fsockopen( string hostname, int port [, int errno [, string errstr [, float timeout]]]) 
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其 中 各 参数 说 明 如 表 13.2 所 示 。 
表 13.2 fsockopen() 函 数 的 参数 说 明 


参数 说 明 | 

hostname 在 Interet 中 使 用 时 ， 参 数 hostname 及 port 分 别 代表 网 址 及 埠 号 。 在 UNIX 下 可 用 作 IPC， | 傣 六 

port hostname 参 数 表示 到 socket 的 路 径 ，port 配 置 为 0 | 一 一 
一 me | 主要 用 于 错误 处 理 ， 可 以 省 略 

timeout 表示 在 指定 的 时 间 内 没有 连接 上 就 中 断 | 


利用 fsockopen0 函 数 连 接 服务 器 后 , 将 返回 一 个 Socket 句柄 ,之 后 通过 文件 操作 函数 ，| 
如 fgets()、fputs()、feof0) 等 接收 消息 或 发 送 命 令 。 本 邮件 系统 将 利用 fsockopen() 接 收 POP3 | 
服务 器 邮箱 中 的 内 容 。 


说 明 : 
fsockopen() 函 数 可 以 与 很 多 服务 器 建立 连接 ， 如 SMTP 服务 器 、POP3 服务 器 等 。 


邮件 发 送 系 统 利 用 fsockopen0) 函 数 连接 POP3 服务 器 ， 代 码 如 下 : 
if(!$this->connection=(@fsockopen(S$this->hostname,$this->port,&$err no,&$err str,$this->timeout)){ 


参数 说 明 如 下 。 

加 ”Sthis->hostname: POP3 服务 器 地 址 。 

加 ”S$this->port: POP3 服务 器 端口 号 。 

加 ”$err no: 系统 连接 POP3 服务 器 发 生 错误 时 返回 的 错误 号 。 
回 $err str: 系统 连接 POP3 服务 器 发 生 错误 时 返回 的 错误 信息 。 
S$this->timeout: 系统 连接 POP3 服务 器 时 的 最 大 超时 时 间 。 


13.7.4 封装 邮件 操作 类 


在 邮件 发 送 系 统 中 ， 将 接收 邮件 的 方法 封装 到 pop3 类 ， 并 且 将 其 单独 存储 于 | 
mailclass.php 中 。 下 面 将 对 该 类 进行 系统 讲解 。 | 

(1) 定义 收 邮件 类 中 的 数据 成 员 。 在 收 邮件 类 pop3 中 ， 首 先 定义 类 中 的 数据 成 员 。 
其 代码 如 下 : 


[al 


01 3 ee ed 
02 class pop3{ 

03 public $hostname=""; //POP3 主机 名 

04 public $port=110; // 主 机 的 POP3 端口 号 ， 一 般 为 110 

05 public $timeout=5; // 连 接 主机 的 最 大 超时 时 间 

06 public $connection=0; // 保 存 与 主机 的 连接 

07 public $state="DISCONNECTED";// 保 存 用 户 登 录 状态 

08 public $debug=0; 1/ 标识 是 否 处 于 调试 状态 ， 如 是 则 输出 调试 信息 
09 public $err_str="; /如 果 出 错 ， 保 存 错 误 信息 

10 public $err no="; /如 果 出 错 ， 保 存 错误 号 
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11 public $resp; /临时 保存 服务 器 的 响应 信息 
12 public $apop; // 指 示 需 要 使 用 加 密 方 式 进行 密码 验证 ， 一 般 服 务 器 不 需要 
13 public $messages: // 保 存 邮 箱 中 邮件 总 数 
14 public $size; // 邮 箱 中 邮件 总 大 小 
15 public $mail list 。“”// 一 维 数组 , 保存 各 个 邮件 的 大 小 及 其 在 POP 服务 器 上 的 序号 
16 public Shead=array0; // 保 存 邮 件 头 内 容 
Lie public Sbody=array0; // 保 存 邮件 内 容 
(2) 定义 收 邮件 类 的 构造 函数 construct0)， 其 代码 如 下 : 
01 Se 
02 function _construct($server,$port=110,$time_out=5){ 
03 S$this->hostname=$server; 
04 S$this->port=$port:; 
05 S$this->timeout=$time_out; 
06 Teturn true; 
07 


注意 : 

上 述 使 用 的 是 PHP5 中 构造 函数 的 定义 方法 ， 而 在 PHP4 中 构造 函数 名 应 该 与 类 名 相 
同 。 当 然 ，PHP5 也 兼容 PHP4 中 构造 函数 的 定义 方法 。 该 构造 函数 的 作用 是 为 数据 成 员 
$hostname、$port、Stimeout 初始 化 。 


(3) 定 义 open 方法 连接 POP3 服务 器 。 在 open 方法 中 ,通过 fsockpen() 函 数 连接 POP3 
服务 器 ， 并 根据 返回 信息 判断 服务 器 是 否 连接 成 功 ， 如 果 成 功 则 设置 标记 服务 器 的 数据 成 
员 $state 的 值 为 AUTHORIZATION。 代 码 如 下 : 


01 
et ace | 
02 function opengO{ 
03 ifS$this->hostname 一 "){f // 判 断 服务 器 名 是 否 为 空 ， 是 则 给 出 提示 信息 
04 S$this->err_str=" 请 输入 主机 名 !"; 
05 return false; 
06 } 
07 if($this->debug) 
08 echo "正在 连接 服务 器 ! "; 
09 /利用 fsockopen0 连 接 POP3 服务 器 
10 if(!$this->connection=(@fsockopen($this->hostname,$this->port,&$err_no,& 
S$err_str,$this->timeout)) { 
11 S$this->err_str=" 连 接 服务 器 失败 , 错误 信息 :".$err_str." 错 误 号 :".$err_no; 
12 return false: 
13 yelse{ 
14 S$this->getresp(): // 利 用 getresp 方法 处 理 服 务 器 端 回 送 的 信息 
5 if($this->debug) // 显 示 返 回信 息 
16 S$this->outdebug($this->resp); 


17 if(substr($this->resp,0,3)!="+OK"){// 如 果 返 回信 息 的 前 3 个 字符 不 为 “+OK” 说 明 服务 器 连接 失败 ， 
并 给 出 错误 信息 
18 S$this->err_ str=" 服 务 器 回 送 无 效 信息 ;".$this->resp." 请 检查 服务 器 是 否 正确 1"; 
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19 return false; 

20 隐 

21 Sthis->state="AUTHORIZATION"; 
22 return true; 


23 } 人 
24 } | 食 - 
和 


收 邮 件 类 判断 某 条 命令 发 送 是 否 成 功 的 方法 是 : 通过 substr0 函 数 提取 出 返回 信息 的 前 3 | 

个 字符 ， 如 果 该 字符 为 “+OK” 表 示 命 令 被 成 功 执行 。 这 是 因为 如 果 POP3 中 某 条 命令 被 成 功 | 

执行 ， 返 回信 息 的 前 3 个 字符 为 “+OK”。 下 面 是 fsockopen0 函 数 连接 成 功 时 返回 的 信息 : | 
+OK Microsoft Windows POP3 Service Version 1.0 <17073156@MRZTS> ready. 


(4) 使 用 getresp 方法 向 POP3 服务 器 接收 命令 。getresp 用 于 取得 服务 器 端 回 送 的 信 
息 ， 并 对 信息 进行 如 下 处 理 : 去 掉 最 后 的 回 车 换行 符 后 ,将 回 送信 息 保存 到 数据 成 员 $resp | 
中 。getresp 方法 代码 如 下 : | 


01 ed 
02 function getrespO{ 

03 for($this->resp="";:){ 

04 if(feof($this->connection)) 

05 return false; 

06 Sthis->resp.=fgets($this->connection,100); /用 fgets0 函 数 接收 命令 
07 Slength=strlen($this->resp); // 获 取 返 回信 息 的 长 度 
08 if(Slength>=2 && substr($this->resp,$length-2,2)=="\\n"){ 

09 S$this->resp=strtok($this->resp,"r\n");“”// 去 掉 回 车 换行 符 

10 return true; 

11 } 

12 } 

13 } 


(5) 使 用 outdebug() 方 法 对 特殊 字符 进行 转换 。outdebusg 方法 的 作用 是 将 调试 信息 
$messages 显示 出 来 ， 并 把 一 些 特殊 字符 进行 转换 及 在 行 尾 加 上 <br>， 这 样 可 使 输出 的 调 | 
试 信息 更 便于 阅读 和 分 析 。outdebug 方法 代码 如 下 : 


01 function outdebug($message){ 
02 echo htmlspecialchars($message)."<br>\n"; 
03 } 


(6) 使 用 command 方法 向 POP3 服务 器 发 送 命令 。 command 方法 可 以 接收 3 个 参数 ，| 
参数 说 明 如 下 : | 
加 ”$command: 发 送 给 服务 器 的 命令 。 
回 ”Sretum_length， 指定 从 服务 器 回 送 信息 中 截取 多 长 作为 命令 回 送 标识 。 
S$retum_code: 指定 标识 的 正确 值 。 | 
如 果 服务 器 回 送 的 第 一 个 字符 为 “+” 则 认为 命令 已 正确 执行 ， 也 可 以 用 前 面 提 到 的 | 
“+OK” 来 判断 。command 方法 代码 如 下 : | 
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起 
01 ds 
02 function command($command,$return length=1,$return code="+"){ 
03 壕 $this->connection 一 0){// 如 果 $this->connection 的 值 为 0， 说 明 还 没有 连接 服务 器 
04 Sthis->err_str=" 没 有 连接 上 任何 服务 器 ， 请 检查 网 络 连 接 !"; 
05 return false; 
06 } 
07 if($this->debug) 1/ 如果 处 于 调试 状态 ， 则 给 出 调试 信息 
08 S$this->outdebug(">>> $command"); 
09 if(!fputs($this->connection,"$Scommand\r\n")){// 如 果 发 送 命 令 失败 则 给 出 信息 
10 Sthis->err_str=" 无 法 发 送 命令 ".$Scommand; 
9 return false; 
12 jelse{ 
13 S$this->getresp0:/ 如 果 发 送 命令 成 功 ， 则 用 getresp0 方 法 来 接收 返回 的 信息 
14 if($this->debug) 
15 Sthis->outdebug($this->resp); 
| 16 if(substr($this->resp,0,$return_length)!=$return_code){ /如 果 返 回 的 信息 不 是 以 “+” 开 头 ， 说 
， 明 返回 的 信息 无 效 
17 Sthis->err_str=$command." 命 令 ， 服 务 器 返回 无 效 : ".$this->resp; 
18 return false; 
19 }else 
20 return true; 
21 } 
多 } 


(7) 使 用 login() 方 法 对 登录 到 POP3 服务 器 上 的 用 户 进行 身份 验证 ,如果 验证 通过 则 
设置 记录 状态 的 数据 成 员 $state 的 值 为 TRANSACTION。 代 码 如 下 : 
01 at pt deh 


02 function login($user,$password){ 
03 ”if($this->state!="AUTHORIZATION"){ // 标 志 状 态 的 值 不 为 AUTHORIZATION， 说 明 还 没 
有 连接 上 服务 器 


04 Sthis->err_str=" 还 没有 连接 到 服务 器 或 状态 不 对 !"; 

05 return false: 

06 

07 这 !$Sthis->apop){ // 不 需要 以 加 密 方式 进行 密码 验证 

08 /向 服务 器 发 送 USER 命令 ， 并 通过 返回 信息 的 前 3 个 字符 是 否 为 “+OK ”来 
验证 命令 是 否 发 送 成 功 

09 if(!$this->command("USER $user",3,"+OK")) 

10 return false: 

11 /向 服务 器 发 送 PASS 命令 ， 并 通过 返回 信息 的 前 3 个 字符 是 否 为 “+OK ”来 
验证 命令 是 否 发 送 成 功 

12 if(!$this->command("PASS $password",3."+OK")) 

13 Teturn false; 

14 Jelse{ // 需 要 以 加 密 方式 进行 密码 验证 

15 /向 服务 器 发 送 APOP 命令 ， 以 加 密 的 方式 进行 身份 验证 

16 这 1!$this->command("APOP $user".md5($this->greeting.$password),3, 
"+OK")) 

17 return false; 

18 } 
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A 
19 Sthis->state="TRANSACTION": /如 果 验 证 通过 则 使 记录 状态 的 数据 成 员 Sstate 的 | 
值 为 TRANSACTION | 
20 return true; 
21 } ! 
! f 
例如 ， 下 面 是 用 户 成 功 登录 服务 器 时 返回 的 信息 。 | 镀 - 
Note 
+OK User successfully logged on Note 


上 面 代 码 中 第 一 个 “+OK” 说 明 USER 命令 发 送 成 功 ,第 二 个 “+OK” 说 明 PASS 命 | 
令 发 送 成 功 。 | 

(8) 使 用 stat() 方 法 获取 邮件 总 数量 及 总 大 小 。stat() 方 法 将 向 服务 器 发 送 STAT 命令 ， | 
从 而 获得 邮件 总 数量 和 总 大 小 。 其 代码 如 下 : | 


01 ee ed Posse mde edn de | 
02 function statO{ 
03 if($this->state!="TRANSACTION"){ 1/ 判断 用 户 是 否 已 经 成 功 登 录 
04 Sthis->err_str=" 还 没有 连接 到 服务 器 或 没有 登录 成 功 !"; 
05 return false: 
06 } 
07 if(!$this->command("STAT",3,"+OK")) // 向 服务 器 发 送 STAT 命令 
08 return false: 
09 else{ 
10 Sthis->resp=strtok($this->resp," "); 1/ 取得 “+OK” 
11 S$this->messages=strtok(" "); // 取 得 邮件 总 数 
12 Sthis->size=strtok(" "); // 取 得 邮件 总 字 节 数 
LE return true; 
14 } 
15 } 
例如 ， 下 面 是 STAT 命令 发 送 成 功 后 返回 的 信息 。 
+OK 3 1504 


上 面 代码 中 ，“+OK” 说 明 命令 发 送 成 功 ，3 表示 邮箱 中 有 3 个 邮件 ，1504 表示 邮件 | 
总 大 小 为 1504 字 节 。 
在 stat0 方 法 中 应 用 strtok0 函 数 对 字符 串 进行 分 割 。strtok0 函 数 的 语法 格式 如 下 : 


string strtok(string strl, string str2) 


上 述 代码 实现 了 从 strl 中 返回 str2 首 字母 前 的 字符 串 。 如 果 某 程序 中 有 多 个 strtok0 | 
函数 ， 只 需要 第 一 个 strtok0 函 数 给 出 strl 的 值 ， 第 二 个 以 后 strl 的 默认 值 为 上 次 返回 值 后 | 
剩余 的 字符 串 。 

(9) 定义 lstmail0 方 法 获取 指定 邮件 的 大 小 。 使 用 listmail0 方 法 获取 指定 邮件 大 小 的 | 
代码 如 下 : 


01 Strip yo i poi 
02 function listmail($mess=NULL.Suni 1d=NULL){ 
03 if($this->state!="TRANSACTION"){ /判断 用 户 是 否 已 经 成 功 登 录 
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04 Sthis->err_str=" 还 没有 连接 到 服务 器 或 没有 登录 成 功 "; 
05 return false; 
| 06 } 
| 07 if($uni id) /如 果 指定 Suni id， 则 使 用 UIDL 命令 
08 $command="UIDL"; 
09 else 
10 Scommand="LIST"; /如果 没 指定 Suni id， 则 使 用 LIST 命令 
11 if($mess) 
12 Scommand.=$mess; ”// 如 果 指 定 Smess， 则 使 用 $mess 所 指定 的 命令 
13 这 !$this->command(Scommand,3,"+OK){ // 利 用 command( 方 法 发 送 命令 
14 return false; 
15 }else{ 
16 $i=0; 
17 Sthis->mail list=array0; /定义 二 维 数 组 ， 用 来 保存 每 个 邮件 的 大 小 
18 Sthis->getresp(): // 获 取 服 务 器 返回 的 信息 
19 while($this->resp!="."){ /为 邮件 结束 的 标志 
20 Si++; 
21 if($this->debug){ 
22 Sthis->outdebug($this->resp); 
23 } 
24 这 $uni id){ 
pe] Sthis->mail list[$i][munm']=strtok($this->resp," "); 
26 S$this->mail list[$i]['size"]=strtok(" "); /获取 某 个 邮件 的 大 小 
27 }else{ 
28 Sthis->mail_list[$i]["num"]=intval(strtok($this->resp," ")); 
29 Sthis->mail list[$i]["size"]=intval(strtok(" ")); 
30 } 
31 S$this->getresp(); 
32 } 
33 return true; 
34 } 
35 } 
| (10) 定义 getmail() 方 法 获取 邮件 头 及 邮件 内 容 。 使 用 getmail(0) 方 法 获取 邮件 头 及 邮 
， 件 内 容 的 代码 如 下 : 
| 01 function getmail($num=1,$line=-1){ 
02 if($this->state!="TRANSACTION"){ // 判 断 用户 是 否 已 经 成 功 登录 
03 Sthis->err_str=" 不 能 收 邮件 ， 还 没有 连接 服务 器 或 登录 成 功 !"; 
04 retum false; 
05 } 
06 if($line<0) // 如 果 $line<0 则 返回 所 有 的 邮件 内 容 
07 $command="RETR $num"; 
08 else // 如 果 $line>=0 则 返回 邮件 前 Snum 行内 容 
| 09 Scommand="TOP $num $line": 
| 10 if(1$this->command("$Scommand",3,"+OK")) // 向 服务 器 发 送 命 令 
| 11 return false: 
| 12 else{ 
| 13 Sthis->getresp(); // 如 果 命 令 发 送 成 功 ， 则 获取 返回 信息 
| 14 Sis_ head=true; 
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while($this->resp!="."){ 
if($this->debug) 
S$this->outdebug($this->resp); 
这 substr($Sthis->resp.0.1) 一 ".") ”//. 号 为 邮件 结束 的 标志 
Sthis->resp=substr($this->resp, 1 ,strlen($this->resp)-1); 
ftrim($this->resp) 一 "") /邮件 头 与 邮件 内 容 之 间 以 空 行 分 隔 


Sis_head=false; 
if($is_head) 
S$this->head[]=$this->resp; // 将 邮件 头 每 一 行 的 内 容 放 到 head[] 
else 
S$this->body[]=$this->resp; // 将 邮件 内 容 的 每 一 行 放 到 body[] 数 组 中 
Sthis->getresp(); 
} 
return true; 
} 
} 
(11) 定义 dele0 方 法 删除 指定 邮件 。 使 用 dele() 方 法 删除 指定 邮件 的 代码 如 下 : 
function dele($num){ 
if($this->state!="TRANSACTION"){ // 判 断 用 户 是 否 已 经 成 功 登 录 
Sthis->err_str=" 不 能 删除 邮件 "; 
return false: 
} 
if(!$Snum){ 
Sthis->err_str=" 删 除 的 邮件 参数 不 对 !"; 
return false; 


} 
if($this->command("DELE $num",3,"+OK")) // 删 除 $num 指定 的 邮件 
Teturn true; 
else 
return false: 
b 


(12) 定义 close0 方 法 断 开 与 服务 器 的 连接 ， 代 码 如 下 。 


01 
02 
03 


function closeO{ //dele 方法 的 作用 是 删除 Snum 指定 的 邮件 
if($this->connection!=0){ 
if($this->state—"TRANSACTION") 


04 $this->command("QUIT",3,"+OK"):; // 向 服务 器 发 送 QUIT 命令 , 从 而 断 开 与 服务 器 的 连接 和 执 


行 DELE 命令 
05 
06 
07 
08 
09 
10 


felose($this->connection); 
S$this->connection=NULL; 
S$this->state="DISCONNECTED": 
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( 喜 " 自学 视频 、 源 程序 : 配套 资源 \mA14\ ) 


FTP (File Transfer Protocol) 是 Internet 上 文件 传输 通用 协议 , 位 于 TCP/IP 
协议 的 应 用 层 。 利 用 该 协议 ， 用 户 可 以 将 一 个 完整 的 文件 从 一 各 计算 机 复制 到 
另 一 各 计算 机 上 。 不 过 , 在 使 用 FTP 传输 文件 之 前 , 需要 先 登 录 FTP 服务 器 (可 
以 通过 已 设置 的 FTP 用 户 名 和 密码 登录 服务 器 )。 本 章 将 向 读者 展现 一 个 多 功 
能 FTP 管理 系统 ， 主 要 实现 本 地 系统 与 远程 FTP 服务 器 之 间 文 件 的 上 传 、 下 载 
和 管理 文件 等 功能 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


遍历 FTP 服务 器 目录 
获取 本 地 文件 的 图 标 

实现 FTP 文件 上 传 、 下 载 

本 地 文件 与 FTP 文件 的 维护 操作 
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14.1 FTP 管理 系统 概述 


14.1.1 “模块 概述 公 / 


FTP 管理 系统 是 一 个 可 以 执行 常用 操作 的 上 传 、 下 载 工 具 ， 通 过 它 用 户 可 以 方便 、 快 
捷 地 访问 FTP 服务 器 , 彼此 独立 地 传输 文件 ， 即 多 个 用 户 在 分 别 执行 各 自 的 上 传 、 下 载 和 | 
维护 等 操作 时 互 不 影响 。 另 外 ，FTP 管理 系统 具有 良好 的 远程 管理 性 ， 用 户 可 以 通过 互联 | 
网 登录 FTP 服务器， 执行 远程 管理 操作 。 | 


14.1.2 ”功能 结构 


FTP 管理 系统 包括 由 根 目录 操作 和 子 目 两 大 基本 模块 构成 。 其 中 根 目 录 操 作 包 括 查看 | 
文件 、 查 看 目录 、 删 除 目录 、 创 建 目 录 和 上 传 文件 等 ， 子 目录 操作 包括 查看 文件 、 查 看 目 | 
录 、 更 改 文件 、 移 动 文件 、 删 除 文件 、 创 建 目录 和 上 传 文件 等 。FTP 管理 系统 的 功能 结构 | 
如 图 14.1 所 示 。 | 


FTP 管 理 系统 


图 14.1 FTP 管理 系统 的 功能 结构 
14.1.3 ”程序 预览 | 
为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 几 个 页 面 运行 效果 | 


图 ， 如 果 想 查看 完整 的 效果 图 ， 参 见 配 书 光盘 源 程序 。 
FTP 管理 系统 的 登录 页 面 和 主页 的 运行 效果 分 别 如 图 14.2 和 图 14.3 所 示 。 
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FTP 时 理 系统 


地址: [127.0.0.1 
用 PP 名 : Pr 


SE :eee 


| 14.2 ”登录 页 面 


ee 
.明日 FTP 管 理 系统 V1.0 


图 14.3 FTP 管理 系统 的 主页 
子 目录 下 的 文件 详细 信息 运行 效果 如 图 14.4 所 示 。 


当前 目录 : /AppServ/Len0402/ 


创建 目录 | 上 传 文件 
文件 名 称 文件 大 小 最 后 修改 时 间 更 改名 称 ”删除 文件 ”移动 文件 
acc_manager_2. php 3.5K 2011-05-03 10:37:29 更 改 删除 移动 


14.4 子 目录 下 文件 详细 信息 


| 单 击 文件 名 称 超 链接 ， 下 载 FTP 文件 的 运行 效果 如 图 14.5 所 示 。 单 击 “ 移 动 ” 超 链 
接 ， 移 动 文件 到 指定 文件 目录 的 运行 效果 如 图 14.6 所 示 。 


将 /AppServ/Len0402/acc_manager_2. php 移 动 到 : 


ad: \Appserv\LenD402 {AppServ/ 


# 请 填写 有 效 路 径 ( 如 C:\Leon\Appserv\) <* 请 填写 正确 的 路 径 。“ 加 /hppserv/" 


yl EE [EF] 


图 14.5 FTP 文件 下 载运 行 效果 图 14.6 FTP 文 件 移动 运行 效果 
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14.2 ”FTP 环境 搭建 


14.2.1 FTP 服务 器 端 概述 


FTP 服务 器 端 是 一 种 可 以 将 计算 机 或 服务 机 设置 成 FTP 服务 器 的 软件 工具 。 通过 FTP | 


软件 ， 用 户 或 其 他 使 用 者 就 能 够 使 用 FTP 协议 ， 通 过 互联 网 或 者 同一 网 络 上 的 任何 一 台 


全 入 


计算 机 与 FTP 服务 器 连接 ， 执 行文 件 或 目录 的 复制 、 移 动 、 创 建 和 删除 等 操作 。 这 里 提 | 
到 的 FTP 协议 是 专门 被 用 来 规定 计算 机 之 间 进 行文 件 传输 的 标准 和 规则 。 正 是 因为 有 了 | 
像 FTP 这 样 的 专门 协议 ， 才 使 得 人 们 能 够 通过 不 同类 型 的 计算 机 ， 使 用 不 同类 型 的 操作 | 


系统 ， 实 现 不 同类 型 文件 的 相互 传递 。 
目前 FTP 软件 的 种 类 繁多 , 其 功能 各 有 千秋 。 Serv-U 是 其 中 的 佼 父 者 , 应 用 最 为 广泛 。 
Serv-U 软件 可 以 提供 以 下 功能 : 
符合 Windows 标准 的 用 户 界面 友好 亲切 ， 易 于 掌握 。 
通过 限制 同一 时 间 最 大 的 用 户 访问 人 数 确保 PC 的 正常 运转 。 
安全 性 能 出 众 ， 在 目录 和 文件 层次 都 可 以 设置 安全 防范 措施 。 
能 够 为 不 同 用 户 提供 不 同 设置 ， 支 持 分 组 管理 数量 众多 的 用 户 。 
可 以 基于 IP 对 用 户 授予 或 拒绝 访问 权限 。 
支持 文件 上 传 和 下 载 过 程 中 的 断 点 续 传 。 
支持 拥有 多 个 IP 地 址 的 多 宿主 站 点 。 
能 够 设置 上 传 和 下 载 的 比率 、 硬 盘 空 间 配 额 、 网 络 使 用 带宽 等 ， 从 而 能 够 保证 用 
户 有 限 的 资源 不 被 大 量 的 FTP 访问 用 户 所 消耗 。 
可 作为 系统 服务 后 台 运 行 。 
可 自由 设置 在 用 户 登录 或 退出 时 的 显示 信息 ， 支 持 具有 UNIX 风格 的 外 部 链接 。 


14.2.2 ”Serv-U 的 安装 和 配置 


为 了 保证 FTP 管理 系统 能 够 正常 运行 并 发 挥 其 应 有 的 功效 , 在 使 用 该 系统 前 应 在 远程 
服务 器 上 搭建 FTP 服务 环境 。 本 系统 选择 Serv-U 来 搭建 FTP 服务 环境 。 下 面 介 绍 FTP 服 
务 环境 的 架设 过 程 。 

(1) 通过 互联 网 下 载 Serv-U FTP Server 安装 包 , 然后 将 其 解压 缩 到 指定 文件 目录 下 ， 
双击 ServUSetup.exe 安装 文件 。 

(2) 在 弹出 界面 中 选择 语言 为 中 文 (简体 )， 单 击 “ 确 定 ” 按 钮 ， 进 入 “安装 向 导 ” 
界面 ， 如 图 14.7 所 示 。 

(3) 单 击 “ 下 一 步 ” 按 钮 ， 进 入 许可 协议 查看 界面 。 选 择 “ 我 接受 协议 ” 单 击 “ 下 
一 步 ” 按 钮 ， 进 入 “选择 目标 位 置 ” 界 面 ， 如 图 14.8 所 示 。 在 该 界面 中 设置 Serv_U 安装 
路 径 ， 然 后 单 击 “ 下 一 步 ” 按 钮 。 

(4) 进入 “准备 安装 ”界面 ， 确 认 无 误 后 单 击 “ 安 装 ” 按 钮 ， 开 始 安装 Serv_U， 如 
图 14.9 所 示 。 


加 加 网 同 风 网 网 


网 网 
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欢迎 使 用 Serv-U 安装 向 导 


即将 安装 serv-u 10.5 到 电脑 上 。 


重要 : 如 果 已 安装 serv-u 比较 旧 的 版 本 ， 并且 未 备份 
数据 文件 ;请 立即 退出 本 安装 进程 并 备份 Serv-u 的 安 
著 目 录 和 数据 巨 录 - 


如 果 安 装 到 现存 的 旧版 本 目录 中 ; 典当 前 的 安 
甘 的 版 本 。 这 将 保留 所 有 当前 的 设 托 。 


术 电 脑 程 语 受 著作 权 法 和 国际 公约 的 保护 。 为 鼓励 今 
后 的 开发 工作 ， 请 勿 表 发 您 的 注册 码 。 谢 谢 ! 


单 击 " 下 一 步 "继续 ， 或 单 击 " 取 消 "退出 安装 . 


图 14.7 “安装 向 导 ” 界 面 


选择 目标 位 置 
将 ssrv-u 安装 到 哪里 ? 


有 。 突 装 问 导 将 把 sav-u 安装 到 以 下 文件 来 中 。 
荐 要 继续 ， 单 击 " 下 一 步 *。 如 果 你 要 这 择 不 同 的 文 并 赤 ， 请 单 击 ;浏览 
D;WppSeryiServ-Ul 浏览 人 


至 少 需要 22.9 Me 的 空 采 磁盘 空间 


图 14.8 “选择 目标 位 置 ” 界 面 


准备 安装 
突 装 向 导 现 在 准备 开始 实 装 servu。 


点 击 "安装 ' 恋 续 安 装 ， 如 果 你 想 要 查看 或 者 更 改 设置 请 点 击 "上 一 步 “。 


目标 位 置 : 
Diwppserwserru 


开始 菜单 文件 夹 : 
Serv-U 


附加 任务 : 
更 多 


创建 桌面 图 标 
将 Serv-u 作为 系统 服务 实 装 


Em 


图 14.9 准备 安装 界面 
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(5) 当 Serv_U 的 所 有 部 分 安装 完毕 后 , 将 弹出 如 图 14.10 所 示 的 “完成 Serv_U 安装 ” 


界面 。 


完成 Serv-U 安装 


安装 向 于 的 安装 。 可 以 通过 选择 已 安装 
的 图 标 来 运行 应 用 程序 。 


点 击 "完成 "退出 安装 
古 动 Sev-u 车 理 控 剩 合 


图 14.10 完成 Serv_U 安装 


(6) 单 击 “ 完 成 ”按钮 ， 这 样 FTP 服务 器 端 软件 Serv-U 安装 完毕 。 


此 时 运行 Serv-U 软件 ， 系 统 会 提示 用 户 设置 相应 的 必要 参数 。 下 面 讲 解 如 何 配置 
Serv-U。 | 


(1) 定位 到 新 添加 的 文件 目录 ， 找 到 可 执行 文件 Serv-U-Trayexe， 双 击 该 文件 即 可 | 
启动 Serv_U。 进 入 控制 管理 台 ( 如 图 14.11 所 示 ) 后 ， 首 先 需要 设置 上 传 。 | 


合营 理 (服务 器 ) | 日 一 采 (本 地 管理 | | 例 对 及 


区 域 详细 信息 


腾 务 器 详细 信息 
刁 --in | 沾 直 至 


“配备 六部 器 记 访问 蚌 刚 
i 
查看 您 许可 证 的 相关 信息 并 进行 配置 、 
“查看 有 关 安 装 程序 的 信息 。 
& SRP 

用 户 “创建 、 修 交 和 考 除 全 喇 肝 户 沙 

创建 、 修 交 和 才 史 用 产 | “创建 、 化 汐 和 昌 丰 是 全 am 

“人 a FP 


六 windows 和 人。 


2 ini. 
“创建 ， 候 竣 并 出 兴 吉事 人 


全 局 性 组 
* 剖 建 、 修 或 和 删 际 全 局 用 户 群 组 


王 组 “ 剖 建 、 修 约 和 粒 语 在 北 扣 应 中 的 全 号 用 户 
创 香 、 信 区 和 出 只 用 产 守 bE 


二 
“吉村 和 政和 台中 在 亲 放 中 的 用 广 和 
“创建 、 修 妆 和 删除 Windows 用 产 降 组 - 


目录 
* 为 域内 所 有 用 户 琵 守 目 好 访问 要 | 
“AR Ee 


[Eb 
| .la 
“维护 域 没 癌 ,反攻 击 ， 宪 到 壳 复 以 及 自主 义 


全 局 目录 


“ 力 服 务 器 内 所 有 用 产 配 天 好 
i es 


天 | 明和 器 限 宙 和 设置 


。 
“ 准 护 区 务 吕 设 置 ， 丰 丰 击 ,密码 恢复 与 定制 标 


务 器 于 置 高 级 FTP 命令 设置 和 行 
SL 和 SSH 证 技 人 及 取 置 加 密 


FE 


图 14.11 管理 控制 台 


(2) 单 击 “ 域 详细 信息 ”， 进 入 “ 域 ”界面 。 如 果 是 首次 设置 域 信息 ， 则 需要 单 击 “ 新 


建 域 ” 按 钮 新 建 域 ， 如 图 14.12 所 示 。 


.3359 


] Ke SS .pp 内 型 模块 精 角 


图 14.12 单 击 “ 新 建 域 ”按钮 设置 域 信息 


| (3) 进入 “ 域 向 导 ” 界 面 ， 需 要 进行 4 步 设 置 ， 即 设置 域名 称 ; 设置 FTP 协议 、 端 
| 口 ; 设置 可 用 人 P 地 址 ; 设置 密码 加 密 方式 。 其 中 设置 FTP 端口 、 密 码 加 密 方式 的 界面 如 
| 图 14.13 和 图 14.14 所 示 。 


| 注意 : 
| 如 果 只 是 在 本 机 上 测试 ， 建 议 不 要 更 改 参数 设置 。 


本 J 


| 
| 
| 
| 
| 
| 
| 
| 
| 
| 
国王” Serv 域 各 导 。 本 人 向导 将 帮助 您 在 文件 服务 器 上 创建 域 国 * Serv.J 域 向 导 。 本 向 导 将 帮助 您 在 文件 服务 器 上 创建 域 。 
| 
可 以 使 用 域 通过 各 种 协 说 提供 对 文件 服务 器 的 访问 。 如 果 当 前 许可 证 不 支持 某 能 够 收复 室 码 〈 使 用 电子 邮件 发 送 用 户 登录 信息 ) 。 使 用 双向 加 密 来 保 
提议 ， 由 内 能 天 合用。 基地 该 代用 的 内 训 反 其 相 和 扩 所 本 而。 候 用 时 向 训 宇 人 建生 总 生成 的 新 放 友 。 
| 密码 加 密 模 式 
| 贺 FTP 和 Explict SSLTLS 21 图 使 用 服务 器 设置 (加 密 : 单 向 加 密 ) 
| ) 单 向 加 密 《 更 安全 ) 
| Implict FTPS (SSL/TLS) 990 局 简单 向 加 记 《 不 太 安全 ) 
| 贺 使 用 SsH 的 SFTP 22 人 由 无 加 密 (不 推荐 ) 
| HTTP 80 更 约 服 务 器 设置 
1 HTTPS (SSL 加 密 的 HTTP)》 443 
| 四 允许 用 户 恢复 密码 
| 
| 
| [El 
图 14.13 FTP 端口 设置 图 14.14 ”FTP 密码 加 密 方式 设置 


(4) 填写 域名 称 ， 然 后 单 击 “下 一 步 ” 按 钮 ， 直 至 设置 完成 。 

(5) 完成 新 建 域 后 ， 返 回 管理 控制 台 。 单 击 “ 添 加 新 用 户 ” 选 项 创建 新 用 户 。 单 如 
| “创建 、 修 改 和 删除 用 户 账户 ” 进入 “用 户 属性 ”界面 ， 如 图 14.15 所 示 。 选 择 “用 户 信 
| 息 ” 选 项 卡 ， 在 其 中 依次 设置 登录 ID、 全 名 、 密 码 、 根 目录 、 管 理 权限 、SSH 公共 密 铀 
| 路 径 、 账 户 类 型 、 说 明 等 选项 ， 然 后 单 击 “ 保 存 ” 按 钮 ， 即 可 完成 用 户 的 添加 。 


< 
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| 必 户 信息 辞 组 [欢迎 消息 | 忆 访问 | 各 录 访 问 | 虚拟 器 径 [ 限 制 和 设置 事 太 | 
| & 用 户 吐 户 信息 指定 接 予 该 账户 的 登录 凭证 和 权限 。 


登录 ID 
Mr | 
| 
和 二 ' 好 
千里 权限 3 会 共 在 情人 ! YY 
无 入 中 这 加 " 引 ET | 
二 BE EEE Note 
未 账户 
到 机 和 和 二 二 。， 避 备用 户 至 要 目录 
电子 邮件 地 址 : 总 是 世 许 登录 
避 用 户 必 令 在 下 一 次 登录 时 更 夏 室 硝 
区 启用 有 k 户 
倍 明 


- 
[| Ww | 


图 14.15 “用 户 属性 ”界面 


说 明 : 

这 里 建议 读者 将 根 目录 设置 为 运行 PHP 环境 的 目录 , 以 便 在 执行 远程 操作 时 更 方便 地 
访问 PHP 服务 器 目录 ， 使 FTP 管理 系统 发 挥 更 大 的 作用 。 笔 者 在 设置 Serv-U 服务 器 端 用 
户 时 ， 将 FTP 访问 的 根 目 录 设 置 成 Appserv 集成 环境 所 在 的 目录 。 


至 此 ，FTP 服务 器 端的 架设 和 参数 设置 介绍 完毕 。 如 需要 对 其 他 参数 进行 设置 ， 建 议 
读者 参考 官方 提供 的 有 关 资 料 。 由 于 篇 幅 限 制 ， 其 他 的 设置 参数 不 能 一 一 列举 。 


14.3 用 户 登 录 


14.3.1 ”用户 登录 功能 概述 


这 里 的 用 户 登录 功能 是 指 FTP 服务 器 端 对 用 户 输入 的 瑟 地址、 用 户 名 等 FTP 连接 信 
息 进 行 验证 。 与 普通 登录 验证 方式 不 同 的 是 ， 用 户 在 登录 页 面 提交 的 验证 信息 需要 与 指定 
服务 器 的 预 设 登录 信息 进行 对 比 ， 如 果 用 户 输入 的 连接 信息 与 预 设 信息 匹配 ， 则 用 户 可 以 
顺利 连接 FTP 服务 器 ， 否 则 将 不 能 连接 FTP 服务 器 。 这 样 就 可 以 使 用 户 安全 地 连接 FTP 
服务 器 。 登 录 页 面 的 运行 效果 如 图 14.16 所 示 。 


”FTP 置 理 系 统 


地 让: [127.0.0.1 


RPS: Pr 


Eb 
[| 


图 14.16 登录 镜面 的 运行 效果 | 


训 人 [Pp 内 副 模 关 精 角 


用 户 登 录 功 能 的 实现 涉及 两 个 核心 文件 ，index.php〈 登 录 表单 页 )、configphp (系统 
连接 公共 文件 )。 


| 14.3.2 登录 页 面 设计 


登录 表单 页 (index.php ) 的 结构 相对 比较 简单 ,主要 包含 连接 FTP 服务 器 的 “IP 地 址 ” 
文本 框 、“ 用 户 名 ”文本 框 、“ 密 码 ” 文 本 框 和 “登录 ”按钮 。 登 录 页 面 中 使 用 的 元 素 如 表 
14.1 所 示 。 


表 14.1 登录 页 面 主要 元 素 


元 素 名 称 说 明 
script "js/check.php" 登录 信息 是 否 为 空 验证 脚本 
text | id="ip"name="ip" 连接 服务 器 IP 地 址 
text id="username"name="username" 连接 用 户 名 


| 连接 用 户 密码 
submit “登录 ”按钮 
在 登录 表单 页 中 ， 通 过 check.js 脚本 中 的 chklogin0 函 数 对 提交 的 登录 信息 进行 验证 。 
在 chklogin0 函 数 中 ， 通 过 获取 ID 值 为 login 的 表单 中 的 元 素来 判断 文本 框 是 否 为 空 。 如 
果 表 单元 素 中 文本 框 为 空 ， 则 将 焦点 定位 到 该 元 素 上 。 其 代码 如 下 : 
01 function chkloginO{ 


02 if(login.ip.value—""){ 

03 alert(" 请 填写 连接 卫 "):login ip.focusO:retum false; /判断 连接 他 文本 框 是 否 为 空 

04 } 

05 if(login.username.value—""){ 

alert(" 请 填写 连接 用 户 名 "):login.username.focus0:return false; /判断 连接 用 户 

名 是 否 为 空 

07 } 

08 if(login.password.value—""){ 

09 alert(" 请 填写 连接 密码 ");login.password.focus(0:return false; /判断 连接 密码 
是 否 为 空 

10 } 

11 login.submitO; /提交 login 表单 

也 


14.3.3 ”系统 连接 公共 文件 


检测 FTP 登录 用 户 信息 的 操作 在 config php 文件 中 完成 。 首 先 获取 表单 提交 的 登录 数 
据 ， 将 其 赋 给 指定 的 SESSION 变量 。 其 代码 如 下 : 


01 <?php 
02 ifisset($S_ POST[login])){ /判断 是 否 以 POST 方式 获取 login 传递 参数 
03 $_SESSION['ip'] = $_ POST['ip"]:; // 获 取 验 证 他 地 址 并 赋 给 SESSION 变量 

04 $_SESSION['"user] = $_ POST[musername']: /获取 验证 用 户 名 并 赋 给 SESSION 变量 
05 $_SESSION['pwd'] =$_POST['password]: /获取 验证 密码 并 赋 给 SESSION 变量 
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B 
07 $ip=$ SESSION['ip]; /传递 SESSION 变量 给 相应 变量 
08 $user =$_ SESSION[mser]: | 
09 S$pwd=$ SESSION[pwd']:; | 
然后 ， 将 获取 的 他 地 址 、 用 户 名 、 密 码 作为 连接 参数 ， 远 程 连接 人 Pp 地 址 指向 的 服务 | 
器 。 如 果 连 接 成 功 则 显示 main php 页 面 的 其 他 内 容 ; 如 果 失 败 则 返回 错误 信息 。 相 关 代 码 | 
if 


01 Slink=@ftp connect($ip); /设置 远程 服务 器 连接 标识 | 
02 Slogin= @ftp login(Slink Suser Spwd): // 连 接 远 程 FTP 服务 器 | 
03 这 !$login){ /判断 用 户 是 否 登 录 成 功 | 
04 echo ' 错 误 ，<a href-"index.php"> 返 回 </a>': /和 否则， 返回 错误 提示 | 
05 exit0; | 
06 } | 
07 ?> 


说 明 : 

本 系统 将 config\config php 文件 作为 FTP 连接 的 公共 文件 。 为 了 避免 重复 书写 FIP 连 | 
接 过 程 ， 在 实现 其 他 FTP 功能 时 ， 只 需 通过 include 语句 调用 该 公共 文件 即 可 完成 与 FTP 
服务 器 的 连接 。 


14.4 根 目 录 操 作 | 


14.4.1 根 目 录 功 能 概述 


用 户 登 录 FTP 管理 系统 后 ， 可 以 遍历 用 户 在 FTP 服务 器 中 指向 的 系统 根 目 录 。 用 户 
可 以 单 击 根 目 录 进 入 次 级 根 目 录 ， 并 可 对 根 目录 执行 操作 。 登 录 FTP 管理 系统 的 运行 效果 | 
如 图 14.17 所 示 。 


Wi 和 上 曰 FTP 利 理 系统 
| ne 


fn 服 关 央 的 闪 作 不 统 是 WTX ”当前 时 间 : 2011 生 509 日 发 “本 本 i 


.旭日 FTP 管 理 系 统 V1.0 


图 14.17 FTP 管理 系统 运行 效果 
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| 其 中 根 目录 的 主要 功能 包括 : 查看 根 目录 名 称 、 删 除 目 录 、 创 建文 件 目 录 、 上 传 文件 
| 到 根 目录 等 。 其 主要 是 在 main.php (系统 管理 页 )、 mkdirphp (创建 文件 目录 页 )、rmkdirphp 
| (删除 文件 处 理 页 ) 等 文件 中 实现 的 。 


全 一 ”14.4.2 遍历 文件 根 目录 


遍历 文件 根 目 录 , 就 是 进入 系统 管理 页 (main php), 应 用 ftp_rawlist0 函 数 结合 foreach 
| 语句 输出 根 目 录 中 的 所 有 文件 目录 名 称 。 单 击 文件 目录 名 称 对 应 的 超 链 接 可 进入 次 级 目 
| 录 ; 如果 想 返回 根 目 录 或 上 级 目录 ， 单 击 “ 返 回 上 级 目录 ” 即 可 。 


1. 遍历 根 目录 名 称 
| 应 用 ftp_ pwd0、ftp_rawlist0 函 数 获取 路 径 名 称 、 目 录 列 表 以 及 目录 下 的 文件 详细 信息 ， 


| 代码 如 下 。 
| 01 ”<?php 
| 02 -$dirmame= ftp pwd($link); // 返 回 当前 目录 名 
| 03 if($dimame!="/){ /判断 当前 目录 中 是 否 存在 / 
| 04 S$dirname = /不 存在 则 将 路 径 名 称 赋值 / 
050 
06 S$flist= ftp_rawlist(Slink, $dirname); /返回 当前 目录 下 详细 列表 
07 S$sdir=arrayO; // 更 改变 量 $sdir 数据 类 型 
08 foreach($flist as $value) { // 循 环 输出 目录 详细 列表 
09 S$array = explode(' ', rtrim($value)): // 将 去 右 空白 的 元 素 以 空格 作为 分 割 符 分 隔 
10 S$alen = count($array); // 获 取 数 组 元 素 个 数 
11 Sp = substr($array[0], 0, 1):; // 截 取 返 回 数组 长 度 
ie 这 $p —'d){ 
| 13 if($array[$alen-1] !=".' && $array[$alen-1] (=".."){ 
| 14 $sdir[] = Sarray[$alen-1]; 
| 15 } 
| 16 } 
| 17 
| TS 


| 将 获取 的 当前 目录 名 称 、 目 录 列 表 以 及 文件 详细 列表 ， 在 显示 页 (main.php〉 中 应 用 
| foreach 语句 循环 输出 。 其 代码 如 下 : 
| 01 <div class="mainleft"> 


| 02 <table border="0" cellpadding="0" cellspacing="0"> 
| 03 <2php 
| 04 /以 下 循环 输出 根 目录 中 的 目录 名 称 列表 
| 05 foreach($sdir as $value) 
| 06 echo'<tr> 
| 07 <td height="10" width="30">&nbsp:&nbsp:</td> 
| 08 <td height="25" width="75" align="center" class="STYLE1"> 
| 09 <a href="main.php?dir=".$dirmame.$value."">'.substr($value,0,12).'</a> 
| 10 ”</td> 
11 <td width=50><a href="rmkdir php?dir=.$dimame.$value."> 有 删除 </a></td> 
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12 <td height="5" width="30">&nbsp:&nbsp:</td> 
13 </tr>"; 
14 ?> 
15 </table> 
16 -</div> 
2. 返回 上 级 目录 
返回 上 级 目录 是 通过 单 击 “ 返 回 上 级 目录 ” 超 链 接 来 实现 的 。 其 实现 原理 是 ， 根 据 
URL 地 址 中 传递 的 目录 路 径 dir, 应 用 ftp_chdir 改变 当前 路 径 为 指定 路 径 。 相 关 代码 如 下 : 
om 
02 iftisset($ GET['dir]){ // 获 得 以 GET 方式 传递 的 URL 参数 
03 ftp_chdir($link, $S_GET['dir]): /改变 当前 路 径 为 指定 路 径 
04  } 
05 iflisset($_GET['up']){ // 判 断 用 户 是 否 单 击 "返回 上 级 目录 " 
06 ftp_cdup(S$link); // 改 变 上 级 目录 为 指定 路 径 
07000 
08 iflisset($ GET['down'])){ /判断 用 户 是 否 单 击 文件 目录 名 称 
09 ftp_get($link, $_ GET['down'], $_GET['down'], FTP_BINARY);// 进 入 下 级 文件 目录 
10 } 
I 


第 1] 生 章 “TFIT 刀 管理 系统 § 


14.4.3 ”删除 文件 目录 
在 系统 管理 页 中 ， 每 一 个 文件 目录 名 称 后 面 都 有 一 个 “删除 ” 超 链接 。 单 击 “ 删 除 ” 


超 链 接 ， 将 文件 目录 名 称 作 为 参数 传递 到 rmkdirphp 文件 ， 完 成 文件 目录 的 删除 操作 。 相 


关 代码 如 下 : 
01 <?php 
02 session start(); /开启 全 局 会 话 
03 $ip=$_ SESSION['ip']; // 将 SESSION 变量 中 的 数据 赋值 给 对 应 变量 
04 $user=$ SESSION['user]; 
05 S$pwd=$ SESSION['pwd']: 
06 S$link= @ftp connect($ip); // 连 接 到 指定 FTP 地 址 
07 ”Slogin = @ftp login($link, $user, Spwd); 。”// 连 接 服 务 器 
08 $file=$ GET['dir].$ GET['file']: // 设 置 删 除 文件 位 置 
09 if(ftp delete($link, $file){ 1/ 判断 文件 是 否 被 删除 
10 echo '<script>alert(" 删 除 文件 成 功 !");history.back0;</script>'; /删除 成 功 提示 
11 jelse{ 
i echo '<script>alert(" 删 除 文件 失败 !");history.back0;</script>'; /删除 失败 提示 
I 
14 ftp close($link); // 关 闭 FTP 连接 
Te 


14.4.4 在 根 目 录 中 创建 新 目录 


二 


击 系统 管理 页 中 的 “创建 目录 ”按钮 ， 跳 转 到 mkdir.php 页 面 ， 填 写 要 在 根 目录 中 


“365。 


训 人 [Pp 内 晶 模 半 精 角 


| 创建 的 新 目录 名 称 ， 然 后 将 数据 提交 到 mkdir_ chkphp 文件 中 ， 即 可 完成 新 目录 的 创建 。 
| 在 mkdirphp 文件 中 ， 完 成 目录 名 称 的 添加 。 其 代码 如 下 : 
| 01 <link hre 合 "css/mkdircss" type="text/css" rel="stylesheet" /> 


食 斤 | | 02 <script src="js/check.js"></script> 
Ee | 03 <form id="mkdir" action="mkdir_ chk.php?dir=" method="post" onSubmit—"retumn chkname();"> 
Note 04 <input 1d="dirname" class="inptl" type="text" name="filename" /> 


05 <input type="submit" value=" 提 交 " /> 

06 ”<input type="reset" value=" 重 置 " /> 

07 </form> 

| 在 mkdir_chk.php 文件 中 ， 获 取 表 单 提交 的 目录 名 称 ， 应 用 ftp_mkdir() 函 数 在 FTP 服 
| 务 器 根 目录 中 创建 新 文件 目录 。 其 代码 如 下 : 


01 <?php 
02 session_ start(); 
03 $ip=$_SESSION['ip]: // 将 SESSION 变量 中 的 数据 赋值 给 对 应 变量 


04 $user=$ SESSION['user]: 
05 S$pwd=$ SESSION['pwd']: 


| 06 ifllempty($_GET["dir"]){ // 判 断 是 否 通 过 GET 方式 获取 文件 目录 路 径 

| 07 Spath=$_GET["dir"]; /将 路 径 名 称 赋值 给 对 应 变量 

| 08 Jyelsef 

| 09 S$path=""; /否则 为 根 目录 

| 0 

| 11 $dirmname=$path.$ POST["filename"]; /指定 创建 文件 路 径 

| 12 S$link= @ftp_ connect($ip); // 连 接 指 定 FTP 地 址 

| 13 ”Slogin = @ftp login($link, $user, Spwd); /连接 FTP 服务 器 

| 14 gresult=ftp mkdir($link, $dirname): // 创 建文 件 目 录 

| 15 if($result) { // 判 断 文 件 目录 是 否 被 创建 

| 16 echo '<script>alert(" 目 录 '.$dirmmame."\n 创建 成 功 ! ");window.close0;;</script>'; /提示 
| 创建 成 功 

| 17 }else { 

| 18 echo '<script>alert(" 目 录 '.$dirmame."\n 创建 成 功 1"):history.back0;</script>';， ”// 提 示 创 
| 建 失败 

| 19 

| 20 ftp close($link): // 关 闭 FTP 连接 */ 

| ZI 


| 14.4.5 ”上 传 文件 到 根 目录 下 


在 系统 管理 页 中 ， 单 击 “ 上 传 文件 ”按钮 ， 跳 转 到 upfile php 页 面 中 ， 选 择 要 上 传 的 
本 地 文件 ， 然 后 将 数据 提交 到 upfile_chk.php 文件 中 ， 即 可 实现 将 本 地 文件 上 传 到 服务 器 
指定 的 文件 夹 下 。 

在 upfile.php 文件 中 ， 创 建 表单 ， 添 加 文件 域 ， 将 本 地 文件 提交 到 upfilechk.php 文件 
中 。 其 代码 如 下 : 


! 01 <form action="upfile_chk.php" method="post" enctype="multipart/form-data"> 
| 02 <input class="inptl" type="file" name="file" /> 
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03 <input class="inptl" type="submit" value=" 提 交 " /> 
04 </form> ! 
在 upfile_chk.php 文件 中 , 获取 表单 中 提交 的 本 地 文件 信息 , 应 用 ftp putO 函 数 将 文件 
上 传 到 服务 器 的 根 目 录 中 。 相 关 代码 如 下 : | 全 六 
i Mee 
02 session start(; // 开 启 全 局 会 话 Note 
03 include_once("config/config.php"); /加 载 公共 连接 文件 | 
04 Sup=$ FILES["file"]["tmp name"]; /获取 上 传 文件 | 
05 $file=$ FILES["file"]["name"]; // 定 义 上 传 文件 名 称 | 
06 if(ftp put($link,$file,$up,FTP BINARY)){ 1/ 上 传 文件 到 服务 器 根 目录 中 | 
07 ”echo '<scripf>alert(" 上 传 'Sfile 成 功 ! ");window.close();</script>'; /提示 上 传 成 功 | 
08 Jelse{ | 
09 ”echo '<scripf>alert(" 上 传 '$filen 失败 ! "):historybackO:</script>': ”// 提 示 上 传 失败 | 
10 } | 
11 ftp close($Slink); /关闭 FTP 连接 | 
2 大 | 


14.5 子 目录 操作 


14.5.1 子 目 录 功 能 概述 


单 击 任意 根 目 录 , 即 可 进入 次 级 目录 , 并 在 右 侧 窗口 中 显示 子 目 录 中 包含 的 所 有 文件 。 
子 目 录 中 部 分 操作 与 根 目 录 操 作 大 同 小 异 ， 但 是 进入 子 目 录 后 ， 不 但 可 以 操作 子 目 录 ， 还 | 
可 以 对 子 目 录 下 的 文件 执行 多 种 操作 。 进 入 子 目 录 的 效果 如 图 14.18 所 示 。 | 


_ 和 有 自 FTP 管 理 系 统 , 
| 站 时 IAN UL SO TONG . . | 


tp 报 务 器 的 锦 作 系统 电 WIY ”当前 时 间 : z011 年 05 月 oo 日 » 六 于 日 录 上传 文件 


当 而 目录 : /AppServ/ 


到 建 日 录 | | 上 作文 
文件 名 称 文件 大 小 最 后 修改 时间 更 可 名 称 “扣除 文件 移动 文件 


ainetdl-AprSare S 10 sos L642 。 2010-t2- 


0847s9 页 了 和 


光 急 程序 rar 56151.o 2z0l0-lz-2l 064135 更 以 出 除 移动 


图 14.18 子 目 录 运 行 效果 


用 户 对 子 目 录 执 行 的 操作 包括 : 查看 子 目 录 文 件 、 删 除 子 目录 、 在 子 目录 中 创建 新 文 
件 目录 、 上 传 文件 到 当前 子 目录 、 更 改 子 目录 下 的 文件 名 称 、 移 动 文件 到 其 他 目录 中 ， 以 | 


Prey UD 


VAC 


| 及 下 载 远程 文件 到 本 地 计算 机 中 。 对 子 目 录 的 操作 ， 主 要 是 在 main php (系统 管理 页 )、 


| 说 明 : 


| Temovefile php.php《〔〈 删 除 文件 处 理 页 )、movefile php《〔〈 移 动 文件 处 理 页 ) 等 文件 中 完成 的 。 


由 于 目录 的 创建 和 上 传 文件 等 功能 与 根 目 录 操 作 大 体 相同 ， 在 此 不 再 黄 述 。 下 面 只 对 


| 子 目 录 中 特有 的 一 些 功 能 进行 讲解 。 


14.5.2 


浏览 子 目录 下 的 文件 


| ”登录 FIP 管理 系统 后 , 单 击 任意 根 目录 即 可 进入 次 级 目录 ,并 在 右 侧 窗口 中 显示 子 目 
| 录 中 包含 的 所 有 文件 。 从 中 可 以 查看 文件 名 称 、 文 件 大 小 、 最 后 修改 时 间 等 ， 还 可 以 对 访 
| 目录 下 的 文件 执行 更 改名 称 、 删 除 和 移动 等 操作 。 在 main php 文件 中 ,遍历 子 目录 中 所 有 
| 文件 相关 信息 的 关键 代码 如 下 。 


01 
02 
03 


<?php 

$dimame = ftp pwd($link) // 返 回 当前 目录 名 

if($dimame !="/){ /判断 当前 目录 中 是 否 存在 
$dimame ="/'; /不 存在 则 将 路 径 名 称 赋 值 / 

} 

S$sfile = ftp_nlist($link, $dirname):; // 返 回 当前 根 目 录 列 表 

if(!S$sfile){ // 判 断根 目录 下 是 否 为 空 
$sfile = array(); // 更 改变 量 $sfile 数据 类 型 

$flist = ftp_rawlist($link, $dirname); // 返 回 当前 目录 下 详细 列表 

?> 


通过 foreach 语句 循环 输出 子 目 录 中 的 详细 内 容 。 其 关键 代码 如 下 : 


01 
02 
03 


| text-align:right:" > 


<table border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor="#BAE8BC"> 
<tr> 
<td height="35" colspan="6" bgcolor="#FFFFFF"><div style=" margin:auto 20px; 


| 04 <input type="button" value="" class="btn3" onclick="javascript: Wopen=open 
| (‘path mkdir.php? 
| 05 dir=<?php echo $dirname; ?>",",'height=220,width=350,scrollbars=n0o");"> 
| 06 <input type="button" class="btn4" onclick="javascript: Wopen=open('‘path_upload.php? 
| 07 dir=<?php echo $dirname; ?>",",'height=220,width=350,scrollbars=no");"></div> 
| 08 </td> 
09 </t> 
10 <tr> 
| 11 <td height="28" align="center" bgcolor="#EFFFFF"> 文 件 名 称 </td> 
| 人 <td align="center" bgcolor="#FFFFFF"> 文 件 大 小 </td> 
13 <td align="center" bgcolor="#FFFFFF"> 最 后 修改 时 间 </td> 
14 <td align="center" bgcolor="#FFFFFF"> 更 改名 称 </td> 
| 15 <td align="center" bgcolor="#EFFFFF"> 删 除 文件 </td> 
| 16 <td align="center" bgcolor 一 "#EFFFFF"> 移 动 文件 </td> 
! La </t> 
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于 | 


18 <2php | 
19 foreach($sfile as $value){ /循环 输出 文件 列表 | 
20 ?> | 
21 <tr> | 
22 <td width="250" height="30" align="center" bgcolor="#FFFFFF"> | 到 办 


23 <a href="#" onclick="javascript: Wopen=open(‘download.php?dir=<?php echo $dirname; ?> ! v 
24 &filename=<?php echo S$value;?>",",height=220,width=350,scrollbars=n0"):;"><?php echo | 
S$value; ?></a> 
25 </td> 
26 <td width="70" height="30" align="center"” bgcolor="#FFFFFF"><?php echo 
round(ftp_size(S$link,$value)/1024,2):?> K 


27 </td> 

28 <td width="200" height="30" align="center" bgcolor="#FFFFFF"> | 
29 <2php echo date(Y-m-d H:i:s',ftp mdtm($link,$value)); ?> | 
30 </td> 

3 <td width="60" height="30" align="center" bgcolor="#FFFFFF"> 


32 <a href="#" onclick="javascript: Wopen=open(‘edit.php?file=<?php echo $value; ?> 
33 &dir=<?php echo $dirmame; ?>',",'height=220,width=350,scrollbars=n0"); "> 更 改 </a> 


34 </td> 

35 <td width="60" height="30" align="center" bgcolor="#FFFFFF"> 

36 <a href="removefile.php?file=<?php echo $value; ?>&dir=<?php echo $dimame;?>"> 有 删除 </a> 
人 </td> 

38 <td width="60" height="30" align="center" bgcolor="#FFFFFF"> 


39 <a href="#" onclick="javascript:Wopen=open('movefile.php?dir=<?php echo $dirname; ?> 
40 &filename=<?php echo $value;?>',",'height=220,width=350,scrollbars=no");"> 移 动 </a></td> 


41 </t> 

42 <?php 

43 } //foreach 语句 结束 
44 3 

45 </table> 


14.5.3 ”远程 文件 下 载 


作为 一 款 FTP 管理 工具 ， 下 载 远 程 服务 器 中 文件 这 一 功能 必 不 可 少 。 与 普通 Web 程 
序 下 载 功能 不 同 ,FTP 服务 器 上 的 文件 并 不 能 直接 通过 单 击 超 链接 形式 下 载 , 因为 FTP 服 
务 器 指定 的 目录 不 一 定 是 Web 服务 器 或 提供 服务 的 环境 目录 。 在 此 需要 通过 ftp_get0 函 数 
来 执行 远程 文件 下 载 操作 。 | 
单 击 文件 名 超 链接 ， 进 入 download.php 文件 下 载 页 面 ， 填 写 正确 的 本 地 下 载 路 径 ， 然 
后 单 击 “ 下 载 ” 按钮 ， 将 数据 提交 到 download.php 文件 中 ， 即 可 将 远程 服务 器 中 的 文件 下 | 
载 到 本 地 。download php 文件 的 代码 如 下 : | 


01 <2php | 
02 session start(); // 开 启 全 局 会 话 | 
03 include once("config/configphp"): // 加 载 公共 连接 文件 | 
04 ifl!lempty($ POST["savepath"])){ // 判 断 是 否 存 在 下 载 路 径 | 
05 $absopath=$ POST["savepath"]; /获取 下 载 路 径 | 
06 $di=$ GET["dir"]: // 获 得 下 载 的 文件 路 径 
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07 $dwname=$ GET["filename"]; // 获 取 下 载 文 件 名 称 

08 $file=$dir.$dwname; // 定 义 下 载 文件 

09 $down=$absopath.$dwname; // 定 义 下 载 路 径 

10 ”if(ftp_get($link,Sdown,$file,FTP_BINARY)) { ”// 判 断 是 否 从 FTP 服务 器 中 下 载 文件 
hi echo '<script>alert(" 文 件 '.$file."n 下 载 成 功 ! ");window.close(O;</script>'; 

12 Jyelse{ 

13 echo '<script>alert(" 文 件 '.$file."\n 下 载 ! "):history.back0;</script>'; 

2; 

15 fip close($link); // 关 闭 FTP 连接 

16 Jelse{ // 如 果 下 载 路 径 不 存在 ， 则 向 用 户 展示 下 载 提示 框 
7 7 二 


18 <script src="js/check.js"></script> 

19 <!-- 省 略 部 分 代码 -> 

20 <form id="download" action="#" method= "post" onsubmit="return chkdown();"> 

21 <input 1d="down" class="iptl" type="text" name="savepath" /> 

2 <p><span style=" color:#FF0000;">* 请 填写 有 效 路 径 (如 C:\Leon\Appserv\)</span></p> 

23 <input class="btn"” type="submit” value=" 下 载 " />&nbsp;&nbsp;<input class="btn" 
type="reset" value=" 重 置 " /> 


24 </form> 

25 <?php 

26 } // 结 束 选择 判断 
27 Ts 


说 明 : 
(1) 远程 文件 下 载 操作 中 ， 路 径 输入 的 form 表单 与 远程 文件 的 下 载 操作 均 存储 于 
down.php 文件 中 。 
(2) 远程 文件 下 载 功 能 中 输入 下 载 路 径 要 按照 路 径 规范 书写 。 如 图 14.19 所 示 ， 定义 
的 就 是 一 个 规范 的 文件 下 载 路 径 。 
到 http://localhost/ftp/ adornioe a so 
Ci\tnp\ 


# 请 填写 有 效 路 径 ( 0C: \Leon\Appserv\) 


[rae 


图 14.19 定义 正确 的 远程 文件 下 载 路 径 
14.5.4 ”文件 移动 


用 户 可 以 选 定 文件 并 将 其 作为 对 象 移动 到 FTP 的 其 他 目录 中 。 单 击 文件 列表 中 的 “ 移 
动 ” 超 链接 ， 进 入 移动 路 径 页 面 movefilephp， 从 中 添加 完整 的 移动 路 径 ， 然 后 单 击 “ 移 
动 ” 按 钮 ,将 数据 提交 到 movefile php, 在 movefile php 中 完成 文件 的 移动 操作 。movefile php 


文件 的 代码 如 下 : 
01 <?php 
02 session start(); // 开 启 全 局 会 话 
03 include once("config/config.php"): // 加 载 公共 连接 文件 
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04 $filename=$ GET["dir"].$_GET["filename"]: // 定 义 文件 路 径 | 

05 Sfname=$ GET["filename"]: // 获 取 文件 名 | 

06 if(empty($ POST["movepath"]){ // 判 断路 径 是 否 存 在 | 
079 | 

08 <form id= "movefile" action="#" method="post" onSubmit="return chkpath();"> | 仿 帮 
09 ”将 <?php echo $filename; ?> 移动 到 : | 窒 一 


wo vote 
11 <center> Note 


12 <input class="inptl" id="path" type="text" name="movepath"> | 
13 </center> | 
14 <p> | 
15 <center> | 
16 <span class="notice">* 请 填写 正确 的 路 径 。" 如 /Appservw/"</span> | 
ji </center> | 
18 ”<p> | 
19 <center> 

20 <input class="btn2" type="submit" value=" 移 动 文 件 "> 

21 </center> 

22 </form> 

23 <?php }else{ 

24 $newpath=$ POST["movepath"].$fhname; // 如 果 存 在 则 执行 移动 操作 | 
25 if(ftp_put($link, $newpath, $filename,FTP_BINARY){ /将 文件 移动 到 指定 文件 目录 | 
26 unlink($filename); /删除 原来 位 置 的 文件 | 
2 echo '<script>alert(" 移 动 '.$file.\n 到 .$_ POST[movepath]. 成 功 ! ");window.closeO;</script>"; | 
28 Jelse{ /提示 移动 成 功 | 
29 echo '<script>alert(" 移 动 '$file.\n 到 .$ POST[movepath].' 失 败 ! "):historyback0O:</scripf>' 

300 > // 提 示 移 动 失败 

31 ftp_close($link); // 关 闭 FTP 连接 

32 }2> 


14.5.5 ”更 改 文件 名 称 


在 移动 文件 过 程 中 ,文件 名 称 冲突 可 能 会 导致 移动 失败 ， 因 此 还 必须 提供 更 改 文 件 名 | 
称 的 功能 。 单 击 文件 列表 中 指定 文件 后 的 “更 改 ” 超 链接 ， 跳 转 到 edit.php 页 面 中 ， 从 中 | 
添加 新 的 文件 名 称 ， 然 后 单 击 “ 更 改 ”按钮 ， 将 数据 提交 到 editphp， 即 可 完成 对 文件 名 | 
称 的 更 改 操作 。edit.php 文件 的 关键 代码 如 下 : 


01 ”<!-- 建 添加 文件 名 称 的 表单 -> | 

02 <table> 

03 ”<!-- 省 略 部 分 代码 --> 

04 <td width="770" align="center"><table border="0" cellpadding="0" cellspacing="0"> 

05 <tr> 

06 <td align="center" class="STYLE1"> | 

07 <form id= "forml" name="forml" method="post" 

08 action="edit.php?dir=<?php echo $ GET['dir].'&ftile=.$_ GET[file] ?>" onSubmit="return 
chkamend():;"> 

09 <p> 原 文件 名 为 : <?php echo $_GET['file]; ?></p> 
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10 <p> 输 入 文件 名 : 

11 <input id="news" ="text" name="new" /> 

Lp2 &nbsp; 

13 <input type="submit" name="Submit" value=" 更 改 " /> 

14 </p> 

15 <p> 注 意 : 要 输入 文件 的 全 名 ， 包 括 扩展 名 </p> 

16 </form> 

1 </td> 

获取 URL 参数 ， 应 用 ftp_rename0 〇 函数 将 文件 名 称 更 改 。 其 代码 如 下 : 

01 <?php 

02 /获取 表单 提交 的 文件 名 称 ， 完 成 对 指定 文件 名 称 的 更 改 操作 

03 session start(; /开启 全 局 会 话 

04 include once("config/config.php"); /加 载 公共 连接 文件 

05 Sfile=$ GET[dir].3_GET['file]; // 获 取 文 件 完整 路 径 

06 if($ POST){ /判断 是 否 存在 以 POST 方式 提交 的 数组 

07 Snew =$_ GET['dir'].$ POST[mew']; // 获 取 新 命名 文件 完整 路 径 

08 Sresult = ftp_rename($link, $file, $new); // 更 改名 称 

09 if($result) // 判 断 是 否 更 改 成 功 

10 { 

11 echo '<script>alert(" 文 件 '$_GET[file] 改名 为 m.$_ POST[mew]n 成 功 
");window.close();</script>"; 

12 } // 提 示 更 改 成 功 

13 else{ 

14 echo '<script>alert(" 文 件 '$_GET['file]."\n 改名 为 m.$ POST[mew'] 失败 
");history.backO";</script>"; 

15 } // 提 示 更 改 失 败 

16 } 

> 


说 明 : 


更 改 文件 名 称 时 必须 添加 扩展 名 ， 如 index.php， 而 不 是 index。 如 果 更 改 时 未 添加 扩 
展 名 ， 可 能 会 导致 程序 无 法 正常 运行 。 


14.6 技术 提炼 


14.6.1 ”FTP 文件 操作 技术 


对 FTP 文件 的 操作 ， 应 用 的 是 PHP 内 置 的 FTP 文件 传输 函数 库 中 的 函数 。 下 面 对 其 
中 常用 的 函数 进行 详细 讲解 。 

(1) ftp_connect0 函 数 

功能 : 建立 一 个 新 的 FTP 连接 ， 如 果 成 功 则 返回 一 个 连接 表示 ， 否 则 返回 false。 该 函 
数 的 语法 格式 如 下 : 


resource ftp_connect( string host [, int port [, int timeout]] ) 
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其 中 参数 说 明 如 下 。 


开头 。 
回 port， 可 选 参数 ，FTP 服务 器 的 端口 ， 默 认 端口 为 21。 


回 host: 必要 参数 ,FTP 服务 器 的 地 址 ， 后 面 不 应 以 斜 线 结尾 ， 前 面 也 不 需要 用 ftp:/ 


回 timeout: 可 选 参数 ， 设 置 网 络 传输 的 超时 时 间 限 制 ， 默认 值 为 90 秒 ， 适 用 于 PHP 


4.2.0 及 以 上 版 本 
例如 ， 应 用 ftp_connectO) 函 数 改变 FTP 服务 器 上 一 个 文件 的 属性 ， 代 码 如 下 。 


01 <?php 
02 $fip server="127.0.0.1"; 


03 ”Slink = ftp connect($ftp_server) or die(" 不 能 连接 FTP 服务 器 "); // 建 立 FTP 连接 


04 ”这 $linlo{ 
05 echo "连接 FTP 服务 器 成 功 "; 
06 


08 > 
(2) ftp login0 函 数 


功能 : 登录 到 FTP 服务 器 ， 如 果 成 功 返 回 twe， 否 则 返回 flse。 该 函数 的 语法 格式 如 下 : 


bool ftp login( resource ftp_stream, string username, string password ) 
其 中 参数 说 明 如 下 。 
回 ”ftp_stream: 必要 参数 ，FTP 连接 的 标识 符 。 
回 userame: 必要 参数 ，FTP 服务 器 的 用 户 名 。 
加 ”password: 必要 参数 ，FTP 服务 器 的 密码 。 
例如 ， 应 用 ftp_login0 函 数 登 录 到 FTP 服务 器 ， 代 码 如 下 。 


01 <?php 
02 Slink=fip connect("127.0.0.1"); 


} 
07 ftp close($link); // 关 闭 FTP 连接 


03 S$login result= ftp login(Slink "mr", "mrsoft"); /登录 FTP 服务 器 
04 if($login resulb { 
05 echo "登录 成 功 "; 
06 }else { 
07 echo "登录 失败 "; 
OE 
09 ftp close($link); // 关 闭 FTP 连接 
10 > 
(3) ftp_chdir0 函 数 
功能 : 在 FTP 服务 器 上 改变 当前 目录 为 指定 目录 , 如 果 成 功 返 回 tue, 否则 返回 false。 
该 函数 的 语法 格式 如 下 : 


bool ftp_chdir( resource ftp_stream, string directory ) 
应 用 该 函数 切换 当前 目录 为 指定 目录 的 代码 如 下 : 


01 ”<?php 
*。373。 
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02 Slink=ftp connect("127.0.0.1"):; // 连 接 指 定 地 址 

03 Slogin result = ftp login($link, "mr", "mrsoft"); // 连 接 FTP 服务 器 
04 ftp_ chdir($link, "myfile"); /改变 目录 

05 echo "当前 目录 为 ".ftp pwd(Slinlg); // 输 出 当前 的 目录 名 
06 ftp close($link); // 关 闭 FTP 连接 

oo 


(4) ftp mdtm() 函 数 


互 


功能 : 检查 指定 文件 的 最 后 修改 时 间 , 并 以 UNIX 时 间 戳 的 方式 返 


。 如 果 发 生 错误 ， 


或 文件 不 存在 则 返回 -1。 某 些 FTP 服务 器 可 能 会 不 支持 这 个 特性 。 该 函数 的 语法 格式 如 下 : 


int ftp mdtm( resource ftp_stream, string remote file ) 
其 中 参数 说 明 如 下 。 
加 ”ftp_stream: 必要 参数 ， 指 定 FTP 连接 的 标识 符 。 
回 remote file: 必要 参数 ， 设 置 FTP 服务 器 的 名 称 。 
应 用 该 函数 检查 文件 修改 时 间 的 代码 如 下 : 


01 <?php 

02 S$file="myfile/mingri.php"; // 设 置 文件 名 称 

03 Slink= ftp connect("127.0.0.1"); 1/ 连接 指定 地 址 

04 $login result = ftp login($link, "mr", "mrsoft"); 1/ 连接 FTP 服务 器 
05 Sresult= ftp mdtm(Slink $file); // 检 查 指 定 文件 的 最 后 修改 时 间 
06 if($result—-1){ // 判 断 查 询 的 文件 是 否 存 在 
07 echo "查询 失败 "; 

08 }else { 

09 echo $file." 的 最 后 修改 时 间 :".date('Y-m-d H:is', Sresulb; 。“// 输 出 最 后 修改 时 间 
ID 

11 ftp_close($link); // 关 闭 FTP 连接 

12 > 


(5) ftp_nlist0 函 数 


功能 : 返回 指定 目录 下 的 文件 列表 , 如 果 成 功 则 返回 指定 目录 下 以 文件 名 组 成 的 数组 ， 


否则 返回 false。 该 函数 的 语法 格式 如 下 : 
array ftp_nlist( resource ftp_stream, string directory ) 
应 用 ftp_nlist0 函 数 返 回 指定 目录 下 的 文件 列表 的 代码 如 下 : 


01 <?php 

02 Slink= ftp connect("127.0.0.1"); // 连 接 指定 地 址 

03 S$login result= ftp login($link, "mr", "mrsoft"); // 连 接 FTP 服务 器 

04 Sresult=ftp nlist($link,™ e"); // 返 回 指定 目录 下 的 文件 列表 
05 print r(Sresult); // 输 出 数组 结果 集 

06 ftp close($link); // 关 闭 FTP 连接 
Or 


(6) ftp_rename() 函 数 


功能 : 将 FTP 服务 器 上 的 一 个 文件 或 目录 改名 , 如 果 成 功 返 回 tue, 失败 则 返回 false。 


该 函数 的 语法 格式 如 下 : 
.374 。 
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bool ftp rename( resource ftp_stream, string oldname, string newname ) 


其 中 参数 说 明 如 下 。 

加 ”ftp_stream: 必要 参数 ，FTP 连接 的 标识 符 
回 “oldname: 必要 参数 ， 旧 的 名 称 。 

回 “newname: 必要 参数 ， 新 的 名 称 。 

应 用 该 函数 实现 文件 或 目录 改名 的 代码 如 下 : 


01 
02 


(7) ftp_get0 函 数 
下 载 FTP 服务 器 上 的 文件 ， 


功能 : 
格式 如 下 : 


<?php 
Slink = ftp_connect("127.0.0.1"); 
Slogin result = ftp_login($link, “mr", "mrsoft"); 
Srenamel = ftp_rename($link, "myfile", "mynewfile"); 
if($renamel){ 
echo "更 改 目录 名 成 功 "; 
jelse{ 
echo "更 改 目录 名 失败 "; 
上 
Srename2 = ftp_rename(S$link, "mrphp", "mingri.php"); 
if($rename2){ 
echo "更 改 文件 名 成 功 "; 
jelse{ 


echo "更 改 文件 名 失败 "; 


} 
ftp_close($link); 
?> 


其 中 参数 说 明 如 表 14.2 所 示 。 


// 将 myfile 文件 目录 重新 命名 


// 判 断 目录 名 称 是 否 被 更 改 


// 判 断 文件 目录 名 称 是 否 被 更 改 


/将 mrphp 文件 重新 命名 
/判断 文件 名 称 是 否 被 更 改 


/关闭 FTP 连接 


bool ftp_get( resource ftp_stream, string local file, string remote file, int mode [, int resumepos] ) 


表 14.2 ”ftp_get() 函 数 的 参数 说 明 


参数 说 明 
ftp_stream 必要 参数 ，FTP 连接 的 标识 符 
local file 必要 参数 ， 保 存 到 本 地 的 文件 名 


remote file 


mode 


必要 参数 ，FTP 服务 器 上 的 文件 名 称 


必要 参数 ， 传 送 模式 :FTP ASCI (文本 模式 ) 或 FTP BINARY (二 进 制 模式 ) 


resumepos 


可 选 参数 ， 仅 适用 于 PHP 4.3.0 以 上 版 本 


应 用 该 函数 下 载 指定 文件 的 代码 如 下 : 


01 
02 
03 
04 


<?php 
$down = "mr.php"; 


Sfile = "myfile/mingri.php"; 
Slink = ftp_connect("127.0.0.1"); 


"375。 


/设置 保存 到 本 地 文件 名 
/建立 FTP 连接 


如 果 成 功 返 回 rue， 否则 返回 false。 该 函数 的 语法 | 
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| 05 S$login result = ftp login($link “mr", “mrsoft"); 
06 if(ftp get($link,$Sdown,$file.FTP BINARY)) { 1/ 判断 是 否 从 FTP 服务 器 中 下 载 文件 


| 07 echo "下 载 $file 成 功 "; 
| 08 Jyelse{ 
会 内 | 09 echo "下 载 $file 失败 "; 
NA | i 
| 11 lose($link): // 关 闭 FTP 连接 
Note ns a 3 


(8) ftp fput0 函 数 
| 功能 : 上 传 一 个 已 经 打开 的 文件 到 FTP 服务 器 ， 如 果 上 传 成 功 ， 返 回 ttre， 否 则 返 
| false。 该 函数 的 语法 格式 如 下 : 
bool ftp_fput ( resource ftp_stream, string remote file, resource handle, int mode [, int startpos] ) 


其 中 参数 说 明 如 表 14.3 所 示 。 


互 


表 14.3_ftp_put(O) 函 数 的 参数 说 明 

参数 说 明 
ftp_ stream | 必要 参数 ，ftp 连接 的 标识 符 
remote file | 必要 参数 ， 本 地 已 经 开打 文件 的 句柄 
handle 必要 参数 ，ftp 服务 器 上 的 文件 名 称 
mode 必要 参数 ， 传 送 模式 参数 : FTP ASCII (文本 模式 ) 或 FTP BINARY (二 进 制 模式 ) 
startpos 可 选 参数 ， 仅 适用 于 PHP 4.3.0 以 上 版 本 

应 用 ftp_fput 函数 上 传 指定 文件 的 代码 如 下 : 


01 -<?php 
02 S$up="soft.php"; /1/ 本 地 上 传 文件 名 
03 S$file="myfile/mrsoft.php"; // 指 定 服务 器 中 目录 名 称 


04 $link = ftp_connect("127.0.0.1"); 
05 Slogin_ result = ftp_login($link, "mr", "mrsoft"); 


06 if(ftp put($link, $file, $up,FTP BINARY)) { /判断 是 否 上 传 文件 到 指定 目录 中 
| 07 echo "上 传 $file 成 功 "; 
| 08 }else{ 
| 09 echo "上 传 $file 失败 "; 
| Dy) 
| 11 ftp close(Slink): // 关 闭 FTP 连接 
2 


| 此外， 本 FTP 管理 系统 中 还 存在 其 他 的 FTP 函数， 由 于 这 些 函 数 并 非 本 章 重点 ， 所 
”以 不 再 一 一 介绍 。 


14.6.2 ”循环 输出 技术 


| 所 有 程序 设计 中 都 包含 循环 控制 语句 ， 用 于 反复 多 次 处 理 问题 。 在 FTP 管理 系统 中 
| 遍历 指定 文件 目录 下 的 内 容 , 应 用 的 是 ftp_nlist0 函 数 ， 其 返回 值 为 数组 ， 所 以 选择 foreach 
| 语句 循环 输出 文件 目录 下 的 内 容 。foreach 语句 的 特点 就 是 操作 数组 ， 其 语法 格式 如 下 。 


区 。376。 


第 14 章 于 JP 管 理 系统 | 


foreach (array_expression as $value) 
statement 


foreach (array_expression as $key => $value) | 优 F 


statement | 


其 中 , 第 一 种 格式 遍历 给 定 的 array_expression 数组 。 每 次 循环 中 ， 当 前 单元 的 值 被 赋 
给 $value 并 且 数 组 内 部 的 指针 向 前 移 一 步 〈 因 此 下 一 次 循环 中 将 会 得 到 下 一 个 单元 )。 | 
第 二 种 格式 实现 功能 与 第 一 种 格式 相同 ， 只 是 在 输出 上 略 有 不 同一 一 元 素 循 环 输出 | 


时 ，Skey 被 作为 指定 元 素 的 键 名 一 同 输出 。 


注意 : 
foreach 语句 不 支持 “@” 来 禁止 错误 信息 。 
例如 ， 在 本 系统 中 应 用 foreach 语句 来 循环 输出 子 目 录 中 文件 相关 信息 的 代码 如 下 。 


01 <2php 

02 $dimame = ftp pwd($link); 

03 if($dimame!="/){ 

04 $dirname .="/"; $flist = ftp_rawlist($link, $dirname); 
05 S$sdir = array0; 

06 foreach($flist as $value){ 


07 S$array = explode(' ', rtrim($value)): 

08 S$alen = count($array); 

09 S$p = substr($array[0], 0, 1); 

10 if($p 一 'd){ 

11 if($array[$alen-1] !=".' && $array[$alen-1] (=".."){ 

12 $sdir[] = $array[$alen-1]; 

13 

14 

re 

16 } foreach($sdir as $value) 

17 echo'<tr> 

18 <td height="10" width="30">&nbsp:&nbsp:</td> 

19 <td height="25" width="75" align="center" class="STYLE1"> 

20 <ahref="mainphp?dir=".$dirname.$value.">'.substr($value,0,12).'</a> </td> | 
21 <td width=50><a hre 人 ="rmkdir.php?dir=".$dirmame.$value."> 删 除 </a></td> | 
22 <td height="5" width="30">&nbsp:&nbsp:</td> | 
23 A/tr>'; | 
24 > 


14.6.3 ”应 用 CSS 设置 屏幕 滚动 条 技术 | 


FTP 管理 系统 的 结构 和 一 般 连 接 数据 库 的 Web 程序 不 同 , 其 在 页 面 返回 的 结果 依据 实 | 
际 目录 中 的 文件 , 而 一 般 连接 数据 库 的 Web 程序 则 是 通过 连接 数据 库 并 应 用 数据 库 查 询 语 
句 返 回 查 询 结 果 ; 连接 数据 库 的 Web 程序 可 以 通过 分 页 来 实现 数据 的 分 段 显示 ， 
而 FTP 管理 系统 则 很 难 实现 这 样 的 分 段 显示 。 面 对 文件 目录 中 数目 众多 的 目录 ，FTP 
a SS 
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| 管理 系统 返回 的 文件 结果 将 会 撑 满 整 个 屏幕 如 何 解决 这 一 问题 呢 ? 可 以 应 用 CSS 相关 技 
| 术 设置 屏幕 的 滚动 条 , 来 实现 文件 目录 下 数据 的 滚动 输出 , 即 当 数目 大 于 所 给 边框 长 度 时 ， 
| 系统 的 返回 结果 将 会 以 滚动 条 的 形式 输出 。 

自动 判断 并 设置 滚动 条 的 CSS 的 关键 代码 如 下 : 


overflow:auto; /设置 自动 判断 并 设置 滚动 条 
0 
| 另外 也 可 以 设置 横向 滚动 条 和 纵向 滚动 条 。 其 语法 格式 如 下 : 
| overflow-x:auto; /设置 横向 自动 判断 并 设置 滚动 条 
overflow-y:hidden: /设置 纵向 自动 判断 并 隐藏 滚动 条 


自动 设置 滚动 条 的 一 般 过 程 如 下 : 
| (1) 设置 div 布局 层 ， 并 为 该 层 设置 高 度 。 相 关 代 码 如 下 : 
01 <div class="content"> 
| 02 ”<! 一 HTML 代码 部 分 -> 

03 </div> 

(2) 在 <head></head> 标 签 中 ， 添 加 <style></style> 属 性 标签 ， 并 class 为 content 的 

CSS 样式 表 设 置 为 自动 调节 高 度 和 宽度 。 相 关 代码 如 下 : 

01 <head> 

02 <style type="text/css"> 

03 . Content{overflow:auto;height:200px;} 

04 </style> 
| 05 <head> 
| 这 样 ， 当 类 为 content 的 div 容器 中 的 内 容 高 度 超过 200px 值 时 ， 在 div 的 边框 会 自动 
| 显示 滚动 条 , 用 户 可 以 通过 拖 搜 滚动 条 来 查看 类 为 content 的 div 中 所 显示 的 全 部 内 容 。 其 
运行 效果 如 图 14.20 所 示 。 
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( 铝 # 自学 视频 、 源 程序 : 配套 资源 \mr\15\ ) 


PDF (Portable Document Format) 是 当前 比较 流行 的 电子 文档 与 电子 表 
格 的 一 种 标准 格式 。 它 提供 了 完善 的 压缩 处 理 ， 无 论 创建 者 创建 的 PDF 文档 使 
用 何 种 字体 、 什 么 样 的 图 片 与 版 式 设计 ， 浏 览 者 都 可 以 通过 免费 的 Adobe 
Reader 对 其 进行 阅读 。 本 章 将 要 开发 的 就 是 一 个 支持 PDF 文档 创建 和 下 载 功 
能 的 模块 。 通 过 本 章 的 学 习 ， 读 者 能 够 学 到 以 下 内 容 。 


WB PHP 国际 化 

MH ”FPDF 的 安装 与 配置 
了 PDF 文档 的 创建 

由 向 PDF 中 插入 图 片 
WI 在 PDF 中 绘制 表格 
WI 在 PDF 中 输出 中 文 
MH PDF 文档 的 下 载 


Dip 内 刘 模块 精 角 


15.1 创建 PDF 文档 概述 


全 7 ”15.1.1 模块 概述 


创建 PDF 文档 模块 的 重点 是 将 用 户 添加 的 文章 生成 PDF 文档 ， 并 且 完 成 PDF 文档 的 
下 载 操作 。 其 中 还 实现 了 文章 添加 、 文 章 浏览 等 辅助 功能 ， 并 且 实现 了 网 站 的 国际 化 《区 
支持 中 文 操作 ， 也 支持 英文 输出 )。 


15.1.2 ”功能 结构 
创建 PDF 文档 模块 的 功能 结构 如 图 15.1 所 示 。 


创建 PDF 文 档 模块 
文章 添加 | 文章 管理 


图 15.1 创建 PDF 文档 模块 功能 结构 图 


15.1.3 ”程序 预览 
| 
| 为 了 使 读者 更 好 地 了 解 创 建 PDF 文档 模块 的 功能 , 下 面 列 出 其 中 一 些 功能 的 运行 效果 
| 图 ， 供 广大 读者 参考 。 


| 
文章 添加 页 面 的 运行 效果 如 图 15.2 所 示 。 
| 
| 


会 二 2 
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ET 


的 学 习 毕 从 的 字 习 方案 ， 谷 千 闪 类 进 认 和 六 对 人 
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| 图 15.2 文章 添加 页 面 运行 效果 
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可。 
图 15.3 文章 浏览 页 面 运行 效果 | 
文章 内 容 浏览 和 PDF 文档 下 载 的 链接 如 图 15.4 所 示 。 
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PDF 文档 的 创 理 和 下 载 


图 15.4 文章 内 容 浏览 和 PDF 文档 下 载 链接 
PDF 文档 下 载 的 运行 效果 如 图 15.5 所 示 。 | 


敬 http://192.168.1.96/TM/pdt/pdf/setup_pdfphp?id=9 - Windowsint-. 一 回 加 
全 图 : 网 rapmaszlealeserwPaypat' -| 分 [x][ 亡 sne 


[区 外 


© Mm 


图 15.5 PDF 文档 下 载 的 运行 效果 
通过 Adobe Reader 阅读 PDF 文档 的 内 容 ， 其 运行 效果 如 图 15.6 所 示 。 
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古人 

山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 , 有 龙 则 灵 。 
斯 是 陋室 ， 惟 吾 德 声 。 苔 痕 上 阶 绿 ， 草 色 入 廉 表 - 
谈 笑 有 鸿 俩 ， 往 来 无 白丁 。 

可 以 调 素 堆 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 无 案 评 之 劳 形 ， 
南阳 诸葛 庐 ， 西 网 子 云 亭 。 和 孔子 云 : “ 何 陆 之 有 ? 


20398x29.697 厘米 < 本 


图 15.6 通过 Adobe Reader 浏览 PDF 文档 的 内 容 
15.2 数据 库 设计 


在 本 模块 中 通过 MySQL 数据 库存 储 用 户 上 传 的 文章 , 然后 以 PDF 格式 将 文章 内 容 下 


| 载 到 本 地 。 
| 15.2.1 创建 数据 库 


在 本 模块 中 ， 创 建 db_pdf 数据 库存 储 用 户 上 传 的 文章 数据 。db_pdf 数据 库 中 只 包含 


一 张 数据 表 : tb_articles。 数 据 库 的 结构 及 数据 表 的 情况 如 图 15.7 所 示 。 


胃 服务 器 : localhost ， 成 数据 库 : db_pdf 


表 操作 记录 数 1 ”类 型 整理 
由 articles 间 贡 圆 芭 便 X 4 MyISAM utf8_unicode_ci 
1 个 表 总 计 4 MylSAM utf8_general_ci 


图 15.7 db_pdf 数据 库 结构 


15.2.2 ”创建 数据 表 


在 tb_articles 数据 表 中 存储 用 户 提交 的 文章 信息 ， 包 括 DD、 标题 、 作 者 、 来 源 、 添 加 
时 间 、 内 容 和 类 别 。 其 结构 如 图 15.8 所 示 。 


服务 器 : localhost ”局 教 据 库 : db_pdf ， 目 去: 耻 _articles 


图 15.8 tb_articles 数据 表 的 结构 
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15.2.3 ”连接 数据 库 | 


本 模块 中 将 连接 数据 库 的 操作 存储 于 connections\conn.php 文件 中 , 设置 数据 库 用 户 名 | 
是 root， 密 码 为 111。conn.php 文件 的 代码 如 下 : 


01 <?php 

02 ”$conn=mysql_connect("localhost","root","111") or die(' 连 接 失 败 :' . mysql_error0);// 连 接 服务 器 
03 ”mysql select_db("db_pdf",$conn) or die ("数据 库 选 择 失 败 :' . mysql_error0); // 选 择 数 据 库 
04 mysql query("set names utf8"); // 设 置 页 面 编 码 格式 

05 > 


15.3 ”模块 国际 化 设置 


本 模块 的 国际 化 只 是 一 个 辅助 功能 ， 目 的 是 让 文章 的 添加 和 浏览 功能 支持 中 文 和 英文 
两 种 语言 。 本 模块 国际 化 设置 的 步骤 如 下 : 
(1) 创建 lang 文件 夹 ， 编 写 lang_ch.php 和 lang_en.php 文件 ， 分 别 定义 中 文 语言 配 
置 和 英文 语言 配置 文件 ， 代 码 参 见 配 套 资源 中 的 内 容 。 
(2) 创建 config.php 文件 ， 获 取 浏 览 器 中 语言 选项 的 值 ， 根据 语言 选项 的 值 调用 
的 语言 配置 文件 。 其 代码 如 下 : 
01 <?php 


02 session_ start(); 
03 ”Slang=$_SERVER['HTTP_ACCEPT LANGUAGE']; /获取 语言 选项 


04 Slang=explode(':'$lang); // 以 “;” 对 字符 串 进行 分 隔 

05 -Slang=$lang[0]; /为 数组 赋值 

06 Slang=explode(',',$lang); // 以 “,” 对 字符 串 进 行 分 隔 

07 Slang=$lang[0]; /为 数组 赋值 

08 switch(Slang){ /根据 获取 到 的 语言 选项 的 值 ， 为 变量 赋值 

09 case "en-us": 

10 Slang="en"; 

11 break: 

2 case "zh-cn": 

13 Slang="ch"; 

14 break; 

15 default : 

16 Slang="en"; 

17 break; 

TS 

19 ifisset($_SESSION[lang])){ | 
20 Slang=$_SESSION['lang']: // 为 SESSION 变量 赋值 | 
21 } | 
22 include "lang/lang $lang.php"; // 调 用 不 同 的 语言 配置 文件 | 
2 ?> | 


(3) 创建 setlang.php 文件 ， 根 据 超 链接 中 传递 的 参数 ， 为 SESSION 变量 赋值 ， 并 且 
调用 不 同 的 操作 页 面 。 其 代码 如 下 : 
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0 <php 
| 02 session start(); 
| 03 switch($ GET[lang']){ // 获 取 超 链接 传递 的 语言 选项 的 值 
| 04 case "en": 
| 05 $_SESSION['lang']="en"; /根据 超 链接 的 值 ， 为 SESSION 变量 赋值 
| 06 break; 
07 case "ch": 
08 $_SESSION['lang']="ch": 
09 break: 
10 default : 
11 $_SESSION['lang']="en": 
12 break; 
3 
1 
15 这 $_GET[link] 一 "addnew"){ /根据 超 链 接 的 值 ， 跳 转 到 不 同 操作 页 面 
16 header("Location:add_data.php"); 
17 jelseif ($_GET["link']=="show"){ 
18 header("Location:look_over.php"); 
20 header("Location:index.php"); 
21 > 
22 ?> 


至 此 ， 本 模块 国际 化 设置 完毕 ， 下 面 讲解 模块 具体 功能 的 开发 。 
15.4 文章 浏 览 


15.4.1 文章 浏览 概述 


| 文章 浏览 功能 默认 输出 英文 格式 的 文章 标题 数据 , 运行 效果 如 图 15.9 所 示 。 当 单 击 文 
| 章 标 题 超 链 接 后 , 将 浏览 到 文章 的 详细 信息 , 并 且 在 详细 信息 页 还 设置 了 PDF 文档 下 载 的 
| 超 链 接 ， 运 行 效果 如 图 15.10 所 示 。 


ES 


| 会 First Page Abstract Jin PHP 2011-05-19 02:35:21 


| 19 jelse{ 


Abstract Jin PHP 2011-05-19 02:35:21 


凶 Article Append | 
Abstract Jia PHP 2011-05-19 02:35:21 


于 Article Manage 
一 Abstract Jim PHP 2011-05-19 02:35-21 


图 15.9 英文 格式 的 文章 列表 
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最 Wnpevwernes 吉文 章 完整 内 容 
XxX Em 
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图 15.10 英文 格式 的 文章 浏览 
15.4.2 ”文章 列表 实现 过 程 


创建 index.php 文件 ， 根 据 选 择 的 语言 不 同 ， 输 出 不 同 的 页 面 和 内 容 。 首 先 ， 通 过 
require_once() 语 句 调 用 不 同 的 包含 文件 ， 完 成 数据 库 连 接 和 语言 配置 ， 然 后 ， 执 行 查询 操 
作 ， 从 数据 表 中 读 取出 指定 的 数据 信息 ， 最 后 ， 在 页 面 中 输出 数据 。 其 完整 代码 如 下 : 


01 <?php 
02 require_once('Connections/conn.php’); // 包 含 数据 库 连接 文件 
03 require_once('config.php"); // 包 含 语言 配置 文件 


04 $query rs="select* from tb articles wherelangver='$rs filter orderby id desc"; /定义 查询 语句 
05 $rs=mysql query($query_Is,$conn) or die(mysql error()); // 执 行 查询 语句 


06 $row rs=mysql fetch assoc($rs):; // 获 取 查 询 结果 
07 ?> 

08 <html xmlns="http://www.w3.org/1999/xhtml"> 

09 <head> 


10 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

11 <title><?php echo $index_title ?></title> 

1 <link href="css/style.css" rel="stylesheet" type="text/css" /> 

13 </head> 

14 <body class="twoColLiqLtHdr"> 

15 <div id="container"> 

16 <div id="header"><img src="images/<?php if($rs filter=="en"){echo "OA 021.jpg";}else 
{echo "OA 02.jpg":}?>" 

17 width="780" height="120" border="0" usemap="#Map" /> 

18 <map name="Map" id="Map"><area shape="rect" coords="709,73,776,95" href="setlang.php?lang= 
ch&link=addnew"/> 

19 <area shape="rect" coords="706,98,772.113" href="setlang.php?lang=en&link=addnew" /> 


</map> 
20 <!-- end #header --></div> 
区 <div id="sidebarl"> 
22 <img src="images/<?php if($rs filter—"en"){echo "OA 061.jpg";}else{echo 


"OA 06jpe":}?>" 
23 width="188" height="396" border="0" usemap="#Map2" /> 
24 <map name="Map2" id="Map2"><area shape="rect" coords="47,110,154,131" 
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href="index.php" /> 
25 <area shape="rect" coords="43,71,163,98" href="add_data.php" /></map> 


26 <!-- end #sidebarl] --></div> 
2 <div id="mainContent"> 
优 六 28 <P><?php echo $index topic; ?> 
ne 29 <?php echo $index_author; ?> 
30 <?php echo $index_comefrom: ?> 
31 <?php echo $index_addtime; ?> 
| 32 </P> 

33 <2php 
34 dof 
= ?><P> 
36 <a href="look over.php?id=<?php echo  $row_rs['id'];?>"><?php echo 

$row_rs['topic']:?></a> 
37 <?php echo $row_rs['author']:?> 
38 < ?php echo $row_rs['comefrom']:?> 
39 <?php echo $row_rs['addtime']:?> 
40 </P> 
41 <2php 
42 }while($row_rs=mysql _fetch_assoc($Srs)); 
43 Iysql_free_result($rs): 
44 ?> 
45 <!-- end #mainContent --></div> 
46 <div id="footer"><img src="images/OA_13-14.jpg" width="780" height="78" /> 
47 <!-- end #footer --></div> 
48 <!-- end #container --></div> 
49 </body> 
50 </html> 


index.php 文件 中 输出 的 内 容 , 会 根据 不 同 语言 ( 中 文 和 英文 ) 传递 的 参数 值 进行 切换 
输出 。 


15.4.3 ”文章 浏览 实现 过 程 


单 击 文章 标题 超 链 接 后 , 将 跳 转 到 look_over.php 文件 中 , 在 此 文件 中 根据 超 链 接 传递 
的 文章 ID 查询 出 该 文章 的 详细 信息 ， 并 输出 文章 的 内 容 。 同 时 在 look overphp 文件 中 创 
建 “ 下 载 PDF 文档 ” 超 链接 ， 链 接 到 setup_ pdfphp， 在 此 文件 中 完成 PDF 文档 的 生成 和 
下 载 操作 。look_overphp 文件 的 关键 代码 如 下 : 


01 <?php 

02 require_once('Connections/conn.php"); // 连 接 数 据 库 

03 require_once('config.php"); /配置 语言 

04 Scolname_rs="1'; // 定 义 变 量 值 

05 if(lisset($_GET['id"]){ /检测 变量 是 否 被 设置 

06 Scolname_rs=(get_magic_quotes_gpc0)?$_ GET['id']:addslashes($_GET['id]); // 
对 数据 格式 进行 转换 
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07 } 
08 Squery_rs=sprintf("select * from tb_articles where id=%s",Scolname_rs); ”// 对 字符 串 进 


09 Srs=mysql_query(Squery_rs,Sconn) or die(mysql_error0); // 执 行 查询 操作 
10 Srow_rs=mysql_ fetch_array(Srs); // 获 取 查 询 结果 集 
11 ?> 

12 <html xmlns="http://www.w3.org/1999/xhtml"> 

13 <head> 


14 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
15 <title><?php echo $index_ title ?></title> | 
16 <link href="css/style.css" rel="stylesheet" type="text/css" /> | 


17 </head> | 
18 <body class="twoColLiqLtHdr"> | 
19 <div id="container"> | 


20 <div id="header"><img src="images/<?php if($rs filter=="en"){echo "OA 021.jpg";} 
else{echo "OA 02.jpg";}?>" | 

21 width="780" height="120" border="0" usemap="#Map" /> 

22 <map name="Map" id="Map"><area shape="rect" coords="708,73,775,95" href="setlang.php?lang 
=ch&link=addnew" /> 

23 <area shape="rect" coords="706,98,772,113" href="setlang.php?lang=en&link=addnew" 


/></map> | 
24 <!-- end #header --></div> | 
25 <div id="sidebar1"> | 
26 <img stc="images/<?php 这 Srs filter—"en") {echo "OA_061jpg'jelsefecho "OA._ 06jpg":;}?>"” | 
27 width="188" height="396" border="0" usemap="#Map2" /> | 
28 <map name="Map2" id="Map2"><area shape="rect" coords="47,111,154,132" | 


href="index.php" /> 
29 <area shape="rect" coords="43,71,163,98" href="add_data.php" /></map> 


49 <!-- end #container --></div> 


30 <!-- end #sidebarl --></div> 
31 <div id="mainContent"> | 
32 <P><?php echo $row _rs['topic']:?></p> | 
EE] <P><?php echo $show_author: ?>: <?php echo $row_rs['author]:?></p> | 
34 <P><?php echo $show_comefrom:?>: <?php echo $row_rs['comefrom'];?></p> | 
35 <P><?php | 
36 Slen=iconv_strlen(Srow_rs['content'],'utf-8'); /对 文章 内 容 长 度 进行 统计 | 
37 if(Slen>100){ | 
38 echo iconv_substr(Srow_rs['content"],0,100,"utf-8")."......"; /对 文 | 
章 内 容 进行 截取 输出 | 
39 jelsef | 
40 echo Srow_rs['content']; // 输 出 文章 内 容 | 
41 | 
42 ?></p> | 
43 <P><?php echo $show_upttime:?>: <?php echo $row_rs['addtime']:?></p> | 
44 <P><a href="setup_pdf.php?id=<?php echo Srow_rs['id']; ?>" | 
45 target="_blank"><?php echo Sshow_down;?></a></p> | 
46 <!-- end #mainContent --></div> | 
47 <div id="footer"><img src="images/OA 13-14.jpg" width="780" height="78" /> | 
48 <!-- end #footer --></div> | 
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</body> 

</html> 

<?php 

mysql_ free_result($rs): 
> 


AI 


Note | 技巧 : 


题 数据 


PDF_Chinese 类 


<?php 


header("Content-Type:text/html:;charset=gb2312"); 


include('pdf/chinese.php"): 
include('Connections/conn.php"); 
class PDF extends PDF_Chinese{ 
function HeaderO{ 
Sthis->SetFont('gb',",10); 
S$this->Write(10,' 文 章 "); 
S$this->Ln(20); 


S$this->Cell(0,10,' 第 '.$this->PageNo0.' 页 "); 


} 

function FooterO{ 
$this->SetY(-15): 
Sthis->SetFont('gb',",10); 

9 


bh 
$colname rs id=$_ GET['id"]:; 


在 对 文章 内 容 进行 统计 和 截取 时 ， 应 用 的 是 iconvy *# 系 列 函 数 。 通过 iconv_strlen() 函 数 
统计 文章 的 长 度 ， 通 过 iconv_substrO) 函 数 对 文章 内 容 进行 截取 。 使 用 iconv_*# 系 列 函数 对 
字符 串 进行 操作 ， 可 以 避免 截取 中 文字 符 串 出 现 乱 码 的 问题 。 如 果 这 里 应 用 strlen() 函 数 统 
计 文 章 长 度 ， 通 过 substr0) 函 数 对 文章 内 容 进行 截取 ， 那 么 截取 的 内 容 就 可 能 会 出 现 乱 码 。 


15.4.4 PDF 文档 的 创建 和 下 载 


PDF 文档 的 下 载 应 用 的 是 超 链接 下 载 方式 ， 而 PDF 文档 的 创建 在 setup_pdf.php 文件 
中 完成 ， 其 根据 超 链接 传递 的 ID 值 ， 从 数据 库 
pb 的 方法 完成 PDF 文档 的 创建 。setup_pdf.php 文件 的 完整 代码 如 下 : 


Ph 查 询 出 指定 文章 的 详细 信息 ， 调 用 


// 设 置 页 面 编码 格式 

// 载 入 中 文 组 件 

// 载 入 数据 库 连 接 文件 
// 继 承 中 文 类 

/定义 方法 设置 标题 
/设置 字体 

// 输 出 标题 内 容 

/换行 


/设置 页 脚 


/获取 指定 数据 的 D 


$query_rs_article=sprintf("select * from tb_articles where id=%s",$colname 1s_id);// 定 义 SQL 语句 
$rs_article=mysql query($query_rs_article,$conn) or die(mysql_error0);// 执 行 SQL 查询 语句 


$row_ rs_article=mysql fetch assoc($rs article); // 获 取 查 询 结 果 
Spdf=new PDF(): // 实 例 化 PDF 类 
$pdf->AddGBFont0: /设置 字体 
S$pdf->Open(); /打开 文件 
S$pdf->AliasNbPages(); // 为 每 页 定义 一 个 别名 
S$pdf->AddPage(); // 分 页 方法 
Spdf->SetFont('gb','B'.20): /设置 字体 


S$pdf->Cell(0,10,iconv("utf-8","gb2312",$Srow_rs_article["topic"]));// 输 出 从 数据 库 中 读 取 的 标 


Spdf->Ln(); // 换 行 
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29 Spdf->SetFont('gb',",10); | 
30 $pdf->Cell(0,10,iconv("utf-8","gb2312",Srow_rs_article['author])); 


31 S$pdf->Ln(); | 

32 S$pdf->SetFont('gb',",10): | 

33 Scontent= iconv('utf-8','gb2312',Srow_rs_article['content]); 。 // 设 置 编码 格式 | 贪 内 
34 Spdf->MultiCell(0,5,$content); /| 输出 内 容 ， 并 且 强 行 终止 | 一 
35 Spdf->Ln(); 

36 Spdf:>AddPage(; // 分 页 Note 
37 S$pdf->Output(iconv("utf-8","gb2312",$row_rs_article['topic"]).'.pdf,true); /生成 PDF 文件 | 

38 ?> | 


说 明 : 
有 关 通 过 FPDF 创建 PDF 文档 的 详细 过 程 参见 15.6 节 。 | 
15.5 文章 添加 

15.5.1 文章 添加 概述 


文章 添加 功能 将 用 户 提交 的 文章 数据 添加 到 指定 的 数据 表 中 ， 其 运行 效果 如 图 15.11 
所 示 。 
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图 15.11 英文 格式 文章 添加 的 效果 
15.5.2 ”文章 添加 实现 过 程 


文章 添加 操作 在 add_data.php 文件 中 完成 。 可 以 将 其 划分 为 两 部 分 : 一 是 通过 表单 提 
交 文 章 数据 ;二 是 获取 表单 提交 的 数据 ， 将 其 添加 到 指定 的 数据 表 中 。 
(1) 创建 表单 的 代码 如 下 : | 


01 <form name="form" method="post" action=""> 


02 <p><?php echo $add data topic: ?>: <input type="text" name="topic" value="" /></p> | 

03 <p><?php echo $add data author; ?>: <input type="text" name="author' value="" /></p> | 

04 <p><?php echo S$add data comefrom: ?>: <input type="text"” name="comefrom" | 
value="" /></p> 

05 <p><?php echo $add data_content; ?>: </p> 


-369s 


~ 了 tp 训 蜡 扩 志 半角 


(x 


08 


|， data lenver ?>: 


| 09 
| 10 


11 
12 


/>&nbsp;&nbsp; 


<textarea name="content" id="textarea" cols="65" rows="6"></textarea> 
<p><?php echo $add data langsp; ?>: 
<input type="radio" name="langver" value="en" />&nbsp:&nbsp:<?php echo S$add_ 


<input type="radio" name="langver" value="ch" /><?php echo $add data lchver ?></p> 
<p><input type="submit" name="Submit" value="<?php echo $add data_submit ?>" 


<input type="reset" name="Reset" value="<?php echo $add data reset ?>" /></p> 
</form> 


(2) 获取 表单 提交 的 数据 ， 将 其 添加 到 指定 的 数据 表 中 。 其 代码 如 下 : 


01 
02 


SS 


11 
12 
13 
14 
ji 
16 
i 


| (96s,96s,96s.96s,96s)"， 


18 
19 
20 
ol 
| 22 
| 2 
24 
| 25 
| 26 
| 27 
| 28 


| 说明 


<?php 

header("Content-type:text/html;charset=utf-8"); 。“// 设 置 页 面 编码 格式 
Tequire_once(Connections/conn .php)); // 连 接 数据 库 
require_once('config php): /配置 语言 
/定义 格式 转换 的 方法 


function GetSQLValueString($theValue, S$theType,$theDefinedValue="",$theNotDefinedValue=""){ 
S$theValue=(!get_magic_ quotes gpc()) ? addslashes($theValue):$theValue; 
switch($theType){ 
case "text"; 
StheValue=($theValue !="")? "". $theValue. "" : "NULL"; 
break: 
} 
return $theValue; 
} 
iflisset($_ POST['Submit])){ 
/获取 表单 中 提交 的 数据 ， 并 完成 对 数据 格式 的 转换 


SinsertSQL=sprintf("insert into tb_articles(topic,author,comefrom,content,langver) values 


GetSQLValueString($_POST['topic'],"text"), 
GetSQLValueString($_ POST['author'],"text"), 
GetSQLValueString($_ POST['comefrom'],"text")， 
GetSQLValueString($_POST['content'"],"text"), 
GetSQLValueString($_POST['langver'],"text") 
); 
// 执 行 添加 语句 ， 将 数据 添加 到 数据 表 中 
SResultl=mysql_query($insertSQL.$conn) or die(mysql_error()); 
echo "<script>alert('$add_data_Suc'); window.location.href="add_data.php';</script>"; 


= 


”由 于 本 模块 中 的 功能 偏 少 , 且 重 点 是 讲解 PDF 文档 的 创建 ， 所 以 这 里 只 选择 文章 浏览 
和 文章 添加 作为 辅助 内 容 。 如 果 读 者 感 兴趣 ， 可 以 在 本 模块 基础 上 进行 二 次 开发 ， 增 加 更 
| 多、 更 好 的 功能 。 
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15.6 技术 提炼 


在 15.4.4 节 中 曾 讲解 了 PDF 文档 的 创建 和 下 载 , 但 只 是 从 整个 模块 的 角度 出 发 , 简单 | 会 内 
地 介绍 了 PDF 文档 创建 和 下 载 所 用 文件 ， 以 及 程序 代码 。 下 面 将 对 FPDF 类 中 的 方法 进行 | 全 一 
系统 讲解 。 


15.6.1 FPDF 概述 


FPDF 是 一 个 免费 的 PDF 操作 类 , 通过 它 可 以 完成 基本 的 PDF 创建 功能 , 并 且 它 还 支 | 
持 中 文 (需要 对 相应 字体 进行 配置 )。FPDF 类 中 常用 的 PDF 操作 函数 包括 : Open0 函 数 、| 
AddPage() 函 数 ，SetFont0 函 数 、Cell0 函 数 和 Output0 函 数 。 

可 以 通过 访问 http://www.fpdf.org/ 来 下 载 FPDF, 然后 将 下 载 的 文件 直接 解压 到 某 一 x | 
件 夹 即 可 。 
15.6.2 ”创建 简单 的 PDF 文档 


通过 FPDF 创建 PDF 文档 ， 首 先 要 实例 化 FPDF 类 ， 然 后 调用 类 中 的 函数 (Open0 函 | 
数 、AddPage() 函 数 、SetFont0 函 数 、Cell0 函 数 和 Output0 函 数 ) 完成 文档 的 创建 。 | 
(1) 创建 FPDF 对 象 的 语法 格式 如 下 : 

$pdf= new FPDF([string page-orientation [, string measure-unit [, string page-format]]]) 
其 中 参数 说 明 如 表 15.1 所 示 。 

表 15.1 创建 FPDF 对 象 的 参数 说 明 
说 明 

设置 PDF 文档 的 排版 。 可 选 值 : P， 表 示 竖 向 ， L， 表 示 横向 
设置 文档 中 位 置 的 计量 单位 。 可 选 值 ，pt， 点 ; mm， 毫 米 ， cm， 厘米 ; in, 英寸 | 
设置 PDF 文 档 纸张 的 类 型 。 可 选 值 :表示 纸张 类 型 的 字符 串 ， 例 如 A4、A5、Letter; | 
一 个 包含 有 两 个 元 素 的 二 维 数组 ， 直 接 指明 纸张 的 大 小 | 
(2) Open0) 函 数 用 于 标示 开始 创建 PDF 文档 。 其 语法 格式 如 下 : 

Spdf->OpenO 
(3) AddPage() 函 数 用 于 为 PDF 文档 添加 一 个 新 页 。 其 语法 格式 如 下 : 

$pdf ->AddPage([string page-orientation]) | 
其 中 ，page-orientation 表示 创建 的 PDF 文档 是 横向 还 是 竖 向 的 ， 与 上 过 创建 对象 时 的 | 
使 用 方法 相同 。 

(4) SetFont(O) 函 数 用 于 设置 当前 使 用 的 字体 。 其 语法 格式 如 下 : 
Spdf->SetFont(string font [, string style [, float size]])): 


参数 说 明 如 下 。 


参数 
age-orientation 


measure-unit 


page-format 
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font: 字体 。 
回 style: 样式 。 其 可 选 值 如 下 ; 如果 没 有 指定 style 的 值 ， 则 默认 为 普通 的 形式 。 
| 人 B: 粗 体 。 
妈 分 I: 斜体 。 
医 靖 | 4 U: 下 滑 线 。 
Note 回 size: 字体 的 大 小 。 如 果 不 指定 ， 则 取 默 认 值 12pt。 
(5) Cell0 函 数 用 于 为 当前 PDF 文档 增加 一 个 单元 格 。 其 语法 格式 如 下 : 
S$pdf->Cell(float w [, float h [, string txt [, mixed border [, int In [, string align [, int fill [, mixed link]]]]I]]) 
其 中 参数 说 明 如 表 15.2 所 示 。 
表 15.2 Cell() 函 数 的 参数 说 明 

参数 说 明 

WwW 表示 单元 格 宽度 

h 表 格 高 度 
格 中 添加 的 文本 字符 串 
表示 单元 格 的 边框 。 设 置 为 0 表示 无 边框 ， 设 置 为 1 表示 边框 ; 或 者 工 表示 左边 框 ，T 表 


bolder | 示 顶 部 边框 ，R 表示 右边 框 ，B 表示 底部 边框 
表示 换行 后 ， 设 置 目前 的 位 置 。 可 选 值 为 ，0 表示 向 右 移动 ，1 表示 定位 到 下 一 行 的 开始 ， 
2 表示 向 下 移动 。 默 认 值 为 0 
align ”| 设置 字符 中 的 对 齐 方式 。 可 选 值 为 工 表示 左 对 齐 ，C 表示 居中 ，R 表示 右 对 齐 。 默 认 值 为 


fill 设置 单元 格 的 背景 。 可 选 值 为 : 0 表示 透明 ; 1 表示 不 透明 。 默 认 值 为 0 
link 设置 一 个 URL 链接 或 者 由 AddLink 函数 添加 的 内 部 链接 

(6) Output0 函 数 用 于 输出 PDF 文档 。 其 语法 格式 如 下 : 

$pdf->Output([string filename [, bool download]]) 

参数 说 明 如 下 。 

回 filename: 要 存储 的 文件 名 。 如 果 不 指 定 文件 名 ， 则 浏览 时 会 在 正 中 直接 打开 。 
| download: 指定 是 否 一 定 要 用 户 下 载 查 看 ， 默 认为 false。 
| 下 面 应 用 上 述 介 绍 的 函数 ， 创 建 一 个 PDF 文档 ， 输 出 一 段 英 文字 符 串 。examplel.php 


| 的 代码 如 下 : 

| 01 <?php 

| 02 ”define(FPDF FONTPATH'font): ”// 定 义 font 文件 夹 所 在 路 径 
| 03 require once('pdf/fpdf.php"); // 包 含 fpdf 类 库 文件 

| 04 S$pdf=new FPDF(P', ‘mm','A4); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 , 单位 为 mm, 纸 
| 张大 小 A4 

| 05 S$pdf->Open(); /开始 创建 PDF 

| 06 Spdf->AddPage(); /增加 一 页 

| 07 S$pdf->SetFont(Courier',T,20): /设置 字体 样式 

| 08 S$pdf->Cell(0,0,mrsoft!"): // 增 加 一 个 单元 格 

| 09 S$pdf->Output|: // 输 出 PDF 到 浏览 器 

| 10 = 


| 运行 结果 如 图 15.12 所 示 。 
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| 实 六 太夫 | 医 httpy//192.168.196/TM/pdf/pdflexamplelphp 入 2 
骨 , 国 昌 图 由] 人 ax 国 二 | 二 | 
! 从 
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[re Es 
图 15.12 简单 的 PDF 文 档 Note 


技巧 : | 

上 面 的 示例 是 将 PDF 文档 的 内 容 在 下 浏览 器 中 输出 。 如 果 将 上 述 代码 中 的 

“Spdf->Output0;” 修 改 为 “$pdf->Output('mrsoft.pdf,true);”, 然后 定义 PDF 文档 名 称 为 mrsoftpdf， 
设置 用 户 下 载 查看 ， 即 可 将 PDF 文档 保存 到 服务 器 并 提示 用 户 下 载 。 


15.6.3 在 PDF 中 插入 图 片 


在 FPDF 中 通过 ImageO 函 数 向 PDF 中 插入 图 片 。 其 语法 格式 如 下 : 
Image(string file, float x, float y, float width, float height) 


参数 说 明 如 下 。 

回 x 和 y: 图 片 所 在 位 置 的 坐标 。 

回 ”width 和 height， 图片 的 宽度 和 高 度 。 

下 面 应 用 ImageO 函 数 向 PDF 文档 中 插入 一 个 图 片 ， 并 且 将 PDF 在 下 浏览 器 中 直接 
输出 。example2.php 的 代码 如 下 : 


01 <2php 

02 define(FPDF FONTPATH'font/); // 定 义 font 文件 夹 所 在 路 径 | 
03 require_once(pdfffpdfphp); /包含 和 pdf 类 库 文件 | 
04 ”S$pdf=new FPDF(P', mm','A4'); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 mm， 纸张 大 小 A4 | 
05 。 $pdf>Open0): // 创 建 PDF | 
06 Spdf->AddPage(): // 增 加 一 页 | 
07 S$pdf->SetFont(Courier,T.20); // 设 置 字体 样式 | 
08 ”S$pdf->Image('images/as.jpg',10,10,80,80); /插入 图 片 | 
09 。 $pdf>Output0: // 输 出 PDF 到 浏览 器 | 
10 ?> ! 


运行 结果 如 图 15.13 所 示 。 


帮 httpy/1921681: 36/TM/pdf/pdffexemple2php - -Windows Interne.. -EN 


| GO ementstpst -|]e|xllPam Pp-| 
| 二 k 夺 = EB hupN192168 1e6 /Tedt/odl/eample2 hp 


则 日 晶 加 | | 
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图 15.13 在 PDF 中 插入 图 片 
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15.6.4 设置 PDF 文档 的 页 眉 与 页 脚 


在 FPDF 中 , 页 眉 与 页 脚 是 通过 重 写 FPDF 类 中 的 Header() 方 法 和 Footer() 方 法 来 实现 


| 的 。 因 为 FPDF 类 中 的 Header() 方 法 和 Footer() 方 法 虽然 存在 ,但 是 方法 体 中 没有 任何 内 容 ， 


| 所 以 设置 PDF 文档 的 页 眉 与 页 脚 需要 重 写 FPDF 类 中 的 Header() 方 法 和 Footer() 方 法 ， 完 
成 页 眉 与 页 脚 的 设置 。 
下 面 创 建 PDF 类 ， 继 承 FPDF 类 ， 在 PDF 类 中 对 Header() 方 法 和 Footer() 方 法 进行 重 

写 ， 完 成 页 眉 与 页 脚 的 设置 。 然 后 ， 创 建 PDF 文档 ， 输 出 一 行 英文 字符 串 和 一 个 图 片 。 
example3.php 的 代码 如 下 : 

01 <?php 

02 define(FPDF FONTPATH'font/); /定义 font 文件 夹 所 在 路 径 

03 require once('pdf/fpdf.php"): // 包 含 和 df 类 库 文件 

04 class PDF extends FPDF{ // 继 承 FPDF 类 ， 重 写 页 眉 与 页 脚 的 方法 

05 function HeaderO{ 1/ 设置 页 眉 

06 Sthis->SetFont('Arial','B',15); ”// 设 置 页 眉 字体 

07 Sthis->Write(10,'Title'); // 写 入 页 收文 字 

08 Sthis->Ln(20); // 换 行 

09 } 

10 function FooterO{ 1/ 设置 页 脚 

11 Sthis->SetY(-15); // 设 置 页 脚 所 在 位 置 

12 Sthis->SetFont('Arial',I,8); /设置 页 脚 字体 

13 Sthis->Cell(0,10,'Page - '.Sthis->PageNo()); /输出 当前 页 码 作为 页 脚 内 容 

14 } 

T5000 

16 ”Spdf=new PDF(P', "mm','A4"); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 mm， 纸 张大 小 A4 

17 S$pdf->OpenO; // 创 建 PDF 

18 S$pdf->AddPage(): // 增 加 一 页 

19 。 $pdf->SetFont(Courier,T,20): // 设 置 字体 样式 

20 。 $pdf>Cell(30.30.MRSOFTI); // 输 出 字符 串 


21 ”S$pdf->Image(images/as.jpg',30,50,80,80); ”// 插 入 图 片 
22 ”S$pdf->Output0; /输出 PDF 到 浏览 器 


运行 结果 如 图 15.14 所 示 。 
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图 15.14 设置 PDF 的 页 眉 和 页 脚 
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$Y 
在 设置 PDF 的 页 眉 与 页 脚 时 ， 引 入 了 Write0、LnO、PageNo0 和 SetY04 个 方法 。 


(1) Write() 方 法 | 
Wirite(0) 方 法 用 于 输出 字符 串 ， 当 到 达 文 档 的 右边 位 置 或 者 遇 到 (wn) 时 ,会 自动 换行 。 | 


其 语法 格式 如 下 : 
Write(float h, string txt [, mixed link]) | | 
Note 
参数 说 明 如 下 。 | 


回 h: 定义 字符 串 的 行列 高 度 。 
加 ”txt: 指定 输出 的 字符 串 。 
回 link: 设置 链接 的 网 页 或 者 AddLink() 方 法 的 标识 符 。 
(2) Ln0 方 法 
Ln() 方 法 用 于 换行 操作 。 其 语法 格式 如 下 : 
Ln([float h]) 
其 中 ， 参 数 h 用 于 设置 行 的 高 度 ， 默 认 值 为 最 后 输出 行 的 高 度 。 
(3) PageNo() 方 法 
PageNo() 方 法 用 于 返回 当前 页 码 。 其 语法 格式 如 下 : 
int PageNo(); 
(4) SetY0 方 法 
SetY() 方 法 用 于 设置 页 脚 的 坐标 。 其 语法 格式 如 下 : 
SetY(float y); | 
其 中 ， 参 数 y 为 页 面 上 的 Y 坐标 ， 单 位 为 mm。 如 果 y 为 负数 ， 则 表示 从 页 面 底部 向 | 
上 的 距离 。 例 如 ， 上 面 的 示例 中 设置 y 的 值 为 -15， 即 页 脚 所 在 位 置 距 页 面 底部 15mm。 ”| 


15.6.5 在 PDF 文档 中 绘制 表格 


通过 FPDF 绘制 表格 的 方法 与 输出 文字 的 方法 类 似 ， 在 输出 文字 时 往往 指定 Cell0 方 | 
法 的 边框 参数 为 0， 即 不 输出 边框 ; 在 绘制 表格 时 往往 将 其 设置 成 大 于 0 的 整数 ， 用 于 表 | 
示 边 框 ， 这 就 是 PDF 文档 中 表格 的 单元 格 。 把 这 些 单元 格 组 合 到 一 起 ， 就 构成 了 表格 。 

下 面 看 一 个 输出 表格 的 示例 example4.php， 其 代码 如 下 : 


01 <?php 

02 define(FPDF FONTPATH'font/): /定义 font 文件 夹 所 在 路 径 
03 require once('pdf/fpdf.php"); // 包 含 pdf 类 库 文件 

04 class PDF extends FPDF{ 

05 function HeaderO{ /设置 页 眉 

06 Sthis->SetFont('Arial','B',15); // 设 置 页 眉 字体 

07 Sthis->Write(10,'Pay Slip'); // 写 入 页 眉 文字 

08 Sthis->Ln(20): /换行 

09 } 

10 function FooterO{ // 设 置 页 脚 


0 L_ 


字符 


VAI 


Sthis->SetY(-15); 


// 设 置 页 脚 所 在 位 置 
/设置 页 脚 字体 
// 输 出 当前 页 码 作为 页 脚 内 容 


} 
S$pdf=new PDF(P', mm, 'A4"); // 创 建新 的 FPDF 对 象 ， 竖 向 放 纸 ， 单 位 为 mm， 纸 张大 小 A4 


Sthis->SetFont('Arial','T'.8); 
Sthis->Cell(0,10,'Page - '.$this->PageNo()); 
Spdf->OpenO; 
Spdf->AddPage(); 


Spdf->SetFont('Arial',",14); 
Sheader=array(Name','Age','Duty',Laborage’); 
$data=array(); 


$data[0] = array('Simon','24',"'employee','5,000.00"); 


Sdata[1] = array('Elaine','25',"Manager','6,000.00"): 

$data[2] = array('Susan','25','employee','4,000.00"):; 

$data[3] = array('David','26'.Manager','8,000.00"); 

S$width=array(40,40,40,40); 

for($i=0;$i<count($header):$i++) 

Spdf->Cell($width[$1],6,$header[$1],1); 

$pdf->Ln(); 

foreach($data as $row){ 
$pdf->Cell($width[0].6.$row[0].1); 
S$pdf->Cell($width[1],6,$row[1],1); 
S$pdf->Cell($width[2],6,$row[2],1); 
S$pdf->Cell($width[3],6,$row[3],1); 
S$pdf->Ln(O; 

bp 

S$pdf->Output(); 

?> 


运行 结果 如 图 15.15 所 示 。 


// 开 始 创建 PDF 
/增加 一 页 
/设置 字体 样式 
/设置 表 头 
/设置 表 体 


/设置 每 列 宽度 
/循环 输出 表 头 


/循环 输出 表 体 


// 输 出 PDF 到 浏览 器 


GE htpy/192168.1.96/TM/pdf/pdf/example4.php - Windows Internet Explorer 一 二 
(ex le httpy//192.168.1.96, p =][x)[P eng pi*| 
合 收 B 闪 起 htp//192168.196/TW/pdi/pdfjexample4.php 
得 目 昌 乌 | Cal LI | | 
| 面 Pay Slip | 
CE 
| | 
E34 
| Name [Laborage 
1 酌 Simon 24 lemployee 5.000.00 
| Elane 25 Manager 5.000 00 | 
Susan 25 lemployee 14.000 .00 
| David 5 Manager 8.000.00 
| 20998x29697 醒 米 “ 
一 加 


图 15.15 在 PDF 中 绘制 表格 


15.6.6 在 PDF 文档 中 输出 中 文 


如 果 应 用 


将 是 再 
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前 面 讲解 的 技术 ,直接 通过 FPDF 在 PDF 文档 中 输出 中 文 ， 那 么 输出 的 中 文 
L 码 。 解 决 此 问题 的 方法 是 应 用 FPDF 提供 的 一 个 中 文 插件 ， 继 承 插件 中 的 
”PDF_Chinese 类 ， 在 PDF 文档 中 输出 中 文 。 


第 15 章 创建 了 DF 文档 模块 一 SS 


首先 下 载 并 解压 此 插件 , 复制 其 中 的 chinese php 文件 , 将 其 存储 于 项 目的 指定 文件 夹 | 
下 ; 然后 通过 include 语句 在 创建 PDF 文档 的 文件 中 载 入 chinese.php; 最 后 创建 PDF 类 ， | 
继承 PDF_Chinese 类 ， 在 PDF 文档 中 输出 中 文 。 示 例 example5.php 的 代码 如 下 : | 


01 <?php 食 和 所 


02 include(pdfrchinese .php); // 载 入 中 文 插件 | 
03 class PDF extends PDF_Chinesef /| 继承 中 文 插件 类 Note 
04 function HeaderO{ // 设 置 页 眉 | 
05 Sthis->SetFont('GB',",10); /设置 字体 | 
06 Sthis->Write(10,'FPDF 中 文 测试 '); // 设 置 页 眉 的 内 容 | 
07 Sthis->Ln(20); // 执 行 换行 操作 | 
08 } | 
09 function Footer0{ // 设 定 页 脚 | 
10 Sthis->SetY(-15); // 设 置 页 脚 的 输出 坐标 | 
ij Sthis->SetFont('GB',",10); // 设 置 字体 | 
12 Sthis->Cell(0,10," 第 '.Sthis->PageNo0.' 页 ); /设置 页 脚 输出 的 内 容 | 
13 } 

4 

15 S$pdf=new PDFO; /创建 PDF 文档 

16 S$pdf:->AddGBFont0:; // 添 加 字体 

17 S$pdf->Open(); /开启 文档 

18 。 $pdf->AliasNbPages0; /为 每 个 页 面 定 义 一 个 别名 

19 S$pdf->AddPage(O; // 添 加 页 

20 S$pdf->SetFont(GB',T,20); /设置 字体 


21 。 $stc=' 山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 ， 有 龙 则 灵 。 斯 是 陋室 ， 惟 吾 德 亏 。 苦 痕 上 阶 绿 ， 
草 色 入 帘 青 。 谈 笑 有 鸿儒 ， 往 来 无 白丁 。 可 以 调 素 琴 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 无 案 读 之 劳 形 。 南 阳 诸 
葛 庐 ， 西 蜀 子 云 享 。 孔 子 云 :“ 何 陋 之 有 ? ”'; 


22 Spdf->Write(10,Sstr); /输出 中 文 
23 S$pdf->OutputO; 
24 ?> 
运行 结果 如 图 15.16 所 示 。 
[二 | 


GO- ple) ppv/192168196T 


“| 1x/P sang p-| 
| 斌 路 天 。 | 荐 htpy/1921681s6/TM/pdfjpdfjexample5php 


图 15.16 在 PDF 中 输出 中 文 | 


注意 : 
在 创建 生成 PDF 文档 的 文件 ( example5.php ) 时 ， 页 面 的 编码 格式 要 设置 成 gb2312; 
如 果 页 面 的 编码 格式 是 utf8， 那 么 即使 继承 了 PDF _ Chinese 类 ， 输 出 的 中 文 仍旧 是 乱码 。 


。397 。 


ge VAI 


15.6.7 PDF 文档 下 载 


| 要 实现 PDF 文档 的 下 载 ,只 需 在 通过 Output0 函 数 输 出 PDF 文档 时 添加 两 个 参数 即 可 。 
会 内 | 第 一 个 参数 定义 下 载 的 PDF 文档 的 名 称 ; 第 二 个 参数 设置 为 mre， 提醒 用 户 下 载 查 看 ， 如 
~ | 果 将 第 二 个 参数 设置 为 false， 那 么 生成 的 PDF 文档 会 直接 存储 于 服务 器 中 ， 不 会 提醒 用 
和 户 下 载 查看 。 
| 下 面 对 示例 example5.php 进行 修改 ， 将 其 生成 的 PDF 文档 命名 为 “程序 测试 .pdf”， 
| 并 且 提 醒 用 户 下 载 查 看 。 修 改 的 关键 代码 如 下 : 
01 <?php 
02 include('pdf/chinese.php"); // 载 入 中 文 插件 
03 /省 略 了 部 分 代码 
04 。 $str=' 山 不 在 高 ， 有 仙 则 名 。 水 不 在 深 ， 有 龙 则 灵 。 斯 是 陋室 ， 惟 吾 德 声 。 苦 痕 上 阶 绿 ， 


草 色 入 帘 青 。 谈 笑 有 鸿儒 ， 往 来 无 白 卫 。 可 以 调 素 琴 ， 阅 金 经 。 无 丝竹 之 乱 耳 ， 无 案 肝 之 劳 形 。 南 阳 诸 
| 葛 访 ， 西 蜀 子 云 享 。 孔 子 云 :“ 何 陋 之 有 ? ”“ 


05 Spdf->Write(10,Sstr); // 输 出 中 文 

06 ”S$pdf->Output( 程 序 测试 .pdf,tre); 

or 
| 修改 后 的 示例 存储 于 example6.php 文件 中 ， 运 行 此 示例 将 输出 如 图 15.17 所 示 的 PDF 
| 文件 下 载 页 面 。 


文件 下 载 


EE 


您 想 打开 或 保存 此 文件 吗 ? 

pa 和 名称。 程序 间 式 pdf 

一 | 类 型 : Adobe Acrobat Document 4.82KB 
未 源 ，192.168.196 


© 者 得 各 克 0 


图 15.17 PDF 文档 下 载 


了 。398 。 


